c - C 中的 IIR 滤波器实现

标签 c embedded digital-filter

我正尝试在 C 中为 FRDMKL25Z 实现 IIR 滤波器木板。我当前的代码如下所示:

#include "Cpu.h"
#include "Events.h"
#include "ADC_1.h"
#include "AdcLdd1.h"
#include "DAC_1.h"
#include "PE_Types.h"
#include "PE_Error.h"
#include "PE_Const.h"
#include "IO_Map.h"


#define NSP 16
static uint16_t DACvalue, ADCvalue;
static LDD_TError Error;
static LDD_TDeviceData *MyDacPtr;
int N=10; // Filter order
double NumCoeff[11]={0.8017, -8.0174, 36.0785, -96.2094, 168.3664, -202.0397, 
                    168.3664, -96.2094, 36.0785, -8.0174, 0.8017};

double DenomCoeff[11]={1.0000, -9.5582, 41.1210, -104.8588, 175.5143, -201.4924,
                    160.6706 , -87.8720, 31.5447, -6.7119, 0.6428};

double Signal[NSP], FilteredSignal[NSP];


int main(void)
{
  /* Write your local variable definition here */
  /*** Processor Expert internal initialization. DON'T REMOVE THIS CODE!!! ***/
    int j, k;
    double y, Reg[NSP];
    PE_low_level_init();
    MyDacPtr = DAC_1_Init(NULL);

    for(j=0; j<NSP; j++) Reg[j] = 0.0; // Init the delay registers
    for(;;){
      for(j=0;j<NSP;j++)
     {  
      for(k=N; k>0; k--) Reg[k] = Reg[k-1];  // Shift the delay register values.

      (void)ADC_1_Measure(TRUE);                /* do conversion and wait for the result */
      (void)ADC_1_GetValue16(&ADCvalue);            /* get the result into value variable */
      Signal[j]=(ADCvalue/65535)*3.5;               /*Convert to volts*/
      Reg[0] = Signal[j];    // The denominator

      for(k=1; k<=N; k++)  Reg[0] -= DenomCoeff[k] * Reg[k];

      y = 0;            // The numerator
      for(k=0; k<=N; k++)y += NumCoeff[k] * Reg[k];
      FilteredSignal[j] = y;

      DACvalue=(FilteredSignal[j]*65535)*3.5; //Convert back to 16 bit 
      Error = DAC_1_SetValue(MyDacPtr, DACvalue);               /* Set DA converter output */
     }
  }

代码的一些细节:

  • 我正在使用飞思卡尔的 Processor Expert (PE) 对嵌入式板进行编程;
  • 使用 MATLAB 生成的 IIR 滤波器系数;
  • 由于设备需要实时过滤数据,所以我从

ADC过滤后直接输出到DAC;

一旦刷入电路板,我就无法从 DAC 端口获得任何输出。

我愿意接受所有建议和调试方法。

最佳答案

这是你做错的地方

  • 你还需要跟踪以前的输出
    • 也许您正在尝试,但有趣的嵌套循环不起作用
  • 分母应用于输出,而不是输入
    • 应用系数时,分母并不是真正的分母;之所以这样称呼,是因为频率响应就像分母一样工作。分母系数乘以历史输出,并从输出中减去它们的总和。 (考虑将所有y(n-1)... y(n-k)项移动到等号的另一边,只留下一侧的过滤器输出)
    • 有趣的分母系数是第一个,它乘以您实际尝试计算的新输出!您将把整个结果除以那个结果,以找出新的输出结果。 (在大多数 IIR 滤波器中,这是 1.0,因此您可以跳过它)。
  • 16 位整数除以 65535 为零!,无需转换为伏特(如其他人所述)。
    • 只在最后才做工程,并且只在你需要的时候做;你没有

这是我的建议,同时尝试保持相同的代码结构...

unsigned int ADCvalue, 
int Reg[NSP]; // use signed values instead of unsigned, and no need for double for history of ADCvalue

for(j=0; j<NSP; j++) Reg[j] = 0; // Init the delayed input registers
for(j=0; j<NSP; j++) FilteredSignal[j] = 0.0; // Init the delayed output registers

for(;;)
{  
     for(k=N; k>0; k--) Reg[k] = Reg[k-1];  // Shift the delay register values.
     for(k=N; k>0; k--) FilteredSignal[k] = FilteredSignal[k-1];

     (void)ADC_1_Measure(TRUE);                /* do conversion and wait for the result */
     (void)ADC_1_GetValue16(&ADCvalue);            /* get the result into value variable */

     Reg[0] = ADCvalue - 0x8000;    // Save the previous inputs samples (and shift the zero value to 0)

     y = 0;
     for(k=0; k<=N; k++) y += NumCoeff[k] * Reg[k];               // The numerator
     for(k=1; k<=N; k++) y -= DenomCoeff[k] * FilteredSignal[k];  // The denominator

     FilteredSignal[0] = y/DenomCoeff[0];

     DACvalue= FilteredSignal[0] + 0x8000;   // shift the zero value back to unsigned, centered at 0x8000
     Error = DAC_1_SetValue(MyDacPtr, DACvalue);               /* Set DA converter output */
}

关于c - C 中的 IIR 滤波器实现,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22765809/

相关文章:

python - 在python中计算指数移动平均线

c - 如何使用 C struct 字符串数组避免缓冲区溢出

c++ - 调试 CUDA 代码

c - 违反约束是否总是导致诊断?

embedded - 验证 IPv6 实现

c++ - 就地重新初始化对象

python - Python/Scipy 是否有 firls() 替代品(即加权、最小二乘、FIR 滤波器设计)?

python - 如何使用 scipy.signal.butter 实现多带通滤波器

c - RSA_private_decrypt 返回 -1

c++ - 用于嵌入式平台的 C++ 中的简单 JSON-RPC?