从零开始动手做Verilog实验--04--11阶FIR滤波器

一、什么是11阶FIR滤波器?

1.FIR滤波器(Finite Impulse Response Filter)的中文名叫做有限长单位冲激响应滤波器

2.它的作用就是对输入的数字信号进行"加工",比如滤除不需要的噪音,或者提取特定频率的信号。

3.看代码更简单

二、代码

1.FIR的代码:

a.构建延迟线(数据缓存):代码利用 tap0 到 tap10 这组寄存器,将输入信号 x 逐级向后传递,从而在内部同时保存了当前及过去10个时刻的采样数据,为滤波计算提供完整的历史数据窗口。

b.利用对称性简化计算:通过 t0 到 t5 的赋值操作,将对称位置的抽头数据(如首尾的 tap0 和 tap10)进行两两相加。这种预处理利用了线性相位FIR滤波器的对称特性,有效减少了后续乘法器的使用数量。

c.采用移位相加代替乘法:在计算累加和 sum 时,代码完全避开了硬件资源消耗极大的乘法运算符,而是通过巧妙的位移(<<)和位拼接({})操作来凑出特定的滤波系数(如 16.875、-11.25 等),极大地优化了硬件面积和运算速度。

拆解 t1 的计算逻辑:

t1 << 4:左移 4 位 ,也就是 乘以 16。

{t17, t17:1}:这是算术右移 1 位(最高位补符号位) ,也就是 乘以 0.5。

{t17, t17, t17:2}:这是算术右移 2 位 ,也就是 乘以 0.25。

{t17, t17, t17, t17:3}:这是算术右移 3 位 ,也就是 乘以 0.125。

把它们加起来:

总系数 = 16+0.5+0.25+0.125=16.875

d.位宽扩展与结果输出:最终的计算结果经过 16 位寄存器 sum 暂存,并通过符号位扩展的方式(将最高位重复拼接)赋值给 16 位输出端口 y,确保了有符号定点数在位宽转换过程中的数值准确性。

c 复制代码
`timescale 1ns / 1ps


module FIR_11(

input clk         ,
input [7:0]x           ,
output reg[15:0] y


    );
    
    reg [15:0]  sum;
    reg [7:0] tap0          ;
    reg [7:0] tap1          ;
    reg [7:0] tap2          ;
    reg [7:0] tap3          ;
    reg [7:0] tap4          ;
    reg [7:0] tap5          ;
    reg [7:0] tap6          ;
    reg [7:0] tap7          ;
    reg [7:0] tap8          ;
    reg [7:0] tap9          ;
    reg [7:0] tap10         ;
    reg [7:0] tap11         ;
                            
    reg [7:0] t0       ;
    reg [7:0] t1       ;
    reg [7:0] t2       ;
    reg [7:0] t3       ;
    reg [7:0] t4       ;
    reg [7:0] t5       ;
    
    
    always @(posedge clk)
    begin
    //-------------------------
        t0<=tap5;
        t1<=tap4+tap6;
        t2<=tap3+tap7;
        t3<=tap2+tap8;
        t4<=tap1+tap9;
        t5<=tap0+tap10;
   
    //------------------
        tap10<=tap9;
        tap9<=tap8;
        tap8<=tap7;
        tap7<=tap6;
        tap6<=tap5;
        tap5<=tap4;
        tap4<=tap3;
        tap3<=tap2;
        tap2<=tap1;
        tap1<=tap0;
        tap0<=x;
    
    //---------------------------------
    
    sum<=(t1<<4)+{t1[7],t1[7:1]}+{t1[7],t1[7],t1[7:2]}+
{t1[7],t1[7],t1[7],t1[7:3]}-(t2<<3)-(t2<<2)+t2-{t2[7],t2[7],t2[7:2]}
+(t3<<2)+t3+{t3[7],t3[7],t3[7:2]}+{t3[7],t3[7],t3[7],t3[7],t3[7:4]}
+{t3[7],t3[7],t3[7],t3[7],t3[7],t3[7:5]}
-t4-{t4[7],t4[7:1]}-{t4[7],t4[7],t4[7],t4[7:3]}
+{t5[7],t5[7:1]}-{t5[7],t5[7],t5[7],t5[7],t5[7],t5[7:5]}
+(t0<<7)-((t0<<2)<<2)-(t0<<2)+{t0[7],t0[7:1]}
+{t0[7],t0[7],t0[7:2]}+{t0[7],t0[7],t0[7],t0[7],t0[7:4]};
    
    
    
    
    y<={sum[15],sum[15],sum[15],sum[15],sum[15],sum[15],sum[15],sum[15:7]};
    
  
    
    end

   
endmodule

仿真测试代码:

c 复制代码
`timescale 1ns / 1ps

module tb_FIR_11;

    reg         clk;
    reg  [7:0]  x;
    wire [15:0] y;

    FIR_11 u_FIR_11 (
        .clk(clk),
        .x(x),
        .y(y)
    );

    initial begin
        clk = 0;
        forever #5 clk = ~clk;
    end

    initial begin
        x = 8'd0;
        
        #20;

        #10 x = 8'd127; 
        #10 x = 8'd0;   
        #10 x = 8'd0;
        #10 x = 8'd0;
        
        #200;

        #10 x = 8'd50;
        #200;

        #10 x = -8'd50; 
        #200;
        #10 x = 8'd100;
        #200;

        repeat (20) begin
            #10 x = $random % 256; 
        end

        #100;
        $stop;
    end


endmodule

三、仿真

相关推荐
坏孩子的诺亚方舟7 天前
FPGA系统架构设计实践15_高云Arora V系列时钟体系
fpga开发·系统架构
FPGA小徐7 天前
入门 CNN 结构全解析|从流程图理论到 FPGA Verilog 硬件实现(含习题带讲解)
fpga开发
FPGA小徐7 天前
FPGA 数字信号处理:并行 FIR 与串行滤波器设计原理、对比与完整 Verilog 实现
fpga开发
Saniffer_SH8 天前
【高清视频】Gen6 服务器还没到,Gen6 SSD 怎么测?Emily 现场演示三种测试环境
人工智能·驱动开发·测试工具·缓存·fpga开发·计算机外设·压力测试
zlinear数据采集卡8 天前
双核架构深度解析:ARM+FPGA如何让数据采集卡实现500Ksps高性能?
arm开发·fpga开发·架构
9527华安8 天前
FPGA实现GTH Transceivers Wizard传输2路视频,基于aurora 8b10b编解码架构,提供4套工程源码和技术支持
fpga开发·gth·aurora 8b10b·transceivers
FPGA小徐9 天前
FPGA 数字信号处理(二):并行 FIR 滤波器的 Verilog 全流程设计与实现
fpga开发
国科安芯9 天前
基于AS32S601ZIT2型抗辐照MCU的商业航天卫星姿态确定与控制系统研究
单片机·嵌入式硬件·安全·fpga开发·架构·risc-v
ALINX技术博客9 天前
【黑金云课堂】FPGA技术教程FPGA基础:I2C 总线通信技术
fpga开发·i2c
Hello-FPGA9 天前
Xilinx KU040 FPGA Camera Link 图像采集
c++·fpga开发