在C中实现简单的高通和低通滤波器

Sla*_*off 7 c signal-processing filter portaudio

尝试使用portaudio记录一些数据,然后使用算法过滤器更改录制的语音,然后播放.我已经验证了很多(来自示例),但我对C很新,我认为在我的过滤器实现中我做了一些愚蠢的事情.

#if LOW_PASS 
{
    float RC = 1.0/(CUTOFF*2*3.14);
    float dt = 1.0/SAMPLE_RATE;
    float alpha = dt/(RC+dt);
    float filteredArray[numSamples];
    filteredArray[0] = data.recordedSamples[0];
    for(i=1; i<numSamples; i++){
        filteredArray[i] = filteredArray[i-1] + (alpha*(data.recordedSamples[i] - filteredArray[i-1]));
    }
    data.recordedSamples = filteredArray;
}
#endif
#if HIGH_PASS
{
    float RC = 1.0/(CUTOFF*2*3.14);
    float dt = 1.0/SAMPLE_RATE;
    float alpha = RC/(RC + dt);
    float filteredArray[numSamples];
    filteredArray[0] = data.recordedSamples[0];
    for (i = 1; i<numSamples; i++){
        filteredArray[i] = alpha * (filteredArray[i-1] + data.recordedSamples[i] - data.recordedSamples[i-1]);
    }
    data.recordedSamples = filteredArray;
}
#endif
Run Code Online (Sandbox Code Playgroud)

当录制的信号试图通过这些过滤器时,我得到以下错误:

*** glibc detected *** ./paex_record: free(): invalid pointer: 0xbfd68600 ***
======= Backtrace: =========
/lib/i386-linux-gnu/libc.so.6(+0x75ee2)[0xb75e2ee2]
./paex_record[0x8048fe5]
/lib/i386-linux-gnu/libc.so.6(__libc_start_main+0xf3)[0xb75864d3]
./paex_record[0x80487f1]
======= Memory map: ========
08048000-0804a000 r-xp 00000000 08:05 2363767    /home/svictoroff/Documents/CompArch/portaudio/examples/paex_record
...
bfd68000-bff1a000 rw-p 00000000 00:00 0          [stack]
Aborted (core dumped)
Run Code Online (Sandbox Code Playgroud)

我真的不确定这里发生了什么.有什么想法吗?在脚本终止时从此处调用Free:

Pa_Terminate();
    if( data.recordedSamples )       /* Sure it is NULL or valid. */
        free( data.recordedSamples );
    if( err != paNoError )
    {
        fprintf( stderr, "An error occured while using the portaudio stream\n" );
        fprintf( stderr, "Error number: %d\n", err );
        fprintf( stderr, "Error message: %s\n", Pa_GetErrorText( err ) );
        err = 1;          /* Always return 0 or 1, but no other return codes. */
    }
    return err;
Run Code Online (Sandbox Code Playgroud)

use*_*631 4

问题是 data.recordedSamples 现在(在 时free())指向在堆栈上分配的结构,而不是在堆上!

既然你有这个指令:

data.recordedSamples = filteredArray;
Run Code Online (Sandbox Code Playgroud)

if( data.recordedSamples )
Run Code Online (Sandbox Code Playgroud)

没有用,因为地址 ID 有效,但不一致:它从未被分配,malloc()并且不在堆上,而是在堆栈上!

当您调用 时free(),该地址很可能指向另一个函数的堆栈。

如果需要,可以将过滤后的数据复制回原始数据recordedSamples,只是不要重新分配该指针。

编辑:

用这个:

for(i = 0; i<numSamples; i++) {
    data.recordedSamples[i] = filteredArray[i];
}
Run Code Online (Sandbox Code Playgroud)