9

尝试使用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

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

*** 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)

我真的不确定这里发生了什么。有什么想法吗?Free 是从脚本末尾处调用的 terminate here:

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;
4

2 回答 2

3

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

既然你有这个指令:

data.recordedSamples = filteredArray;

if( data.recordedSamples )

没有用,因为地址 id 有效,但不一致:它永远不会被分配,malloc()它不在堆上,而是在堆栈上!

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

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

编辑:

用这个:

for(i = 0; i<numSamples; i++) {
    data.recordedSamples[i] = filteredArray[i];
}
于 2012-12-14T16:08:54.893 回答
1

看起来您正在尝试释放堆栈变量。您唯一需要调用的时间free是您之前调用过malloc(或它的朋友之一calloc),或者当您调用的库函数的文档说您需要释放它返回的指针时。

顺便说一句,任何时候你释放一个指针,一个好的做法是在之后立即将它设置为 NULL。

堆栈变量一旦超出范围就会消失。可能会帮助您更好地理解。

于 2012-12-14T16:11:32.250 回答