該項(xiàng)目介紹了如何使用 Verilog 實(shí)現(xiàn)具有預(yù)生成系數(shù)的簡(jiǎn)單 FIR 濾波器。
緒論
不起眼的 FIR 濾波器是 FPGA 數(shù)字信號(hào)處理中最基本的模塊之一,因此了解如何將具有給定抽頭數(shù)及其相應(yīng)系數(shù)值的基本模塊組合在一起非常重要。因此,在這個(gè)關(guān)于 FPGA 上 DSP 基礎(chǔ)實(shí)用入門的教程中,將從一個(gè)簡(jiǎn)單的 15 抽頭低通濾波器 FIR 開始,在 Matlab 中為其生成初始系數(shù)值,然后轉(zhuǎn)換這些值用于編寫 Verilog 模塊。
有限脈沖響應(yīng)或 FIR 濾波器定義為脈沖響應(yīng)在特定時(shí)間段內(nèi)穩(wěn)定為零值的濾波器。脈沖響應(yīng)穩(wěn)定到零所花費(fèi)的時(shí)間與濾波器階數(shù)(抽頭數(shù))直接相關(guān),濾波器階數(shù)是 FIR 的基礎(chǔ)傳遞函數(shù)多項(xiàng)式的階數(shù)。FIR 的傳遞函數(shù)不包含反饋,因此如果輸入一個(gè)值為 1 的脈沖,然后輸入一串零值,輸出將只是濾波器的系數(shù)值。
濾波器的作用基本都是用于信號(hào)調(diào)節(jié),主要集中在選擇濾除或允許通過哪些頻率。最簡(jiǎn)單的例子之一是低通濾波器,它允許低于某個(gè)閾值(截止頻率)的頻率通過,同時(shí)大大衰減高于該閾值的頻率,如下圖所示。
該項(xiàng)目的主要重點(diǎn)是在 HDL(具體為 Verilog)中實(shí)現(xiàn) FIR,它可以分解為三個(gè)主要邏輯組件:一個(gè)循環(huán)緩沖器,用于將每個(gè)樣本計(jì)時(shí)到適當(dāng)?shù)乜紤]了串行輸入的延遲、每個(gè)抽頭系數(shù)值的乘法器以及每個(gè)抽頭輸出的求和結(jié)果的累加器。
由于本項(xiàng)目專注于 FPGA 邏輯中 FIR 的設(shè)計(jì)機(jī)制,所以只是使用 Simulink 中的 FDA 工具和 Matlab 為低通濾波器插入一些簡(jiǎn)單參數(shù),然后使用生成的系數(shù)值放到 Verilog 模塊中完成濾波器的設(shè)計(jì)(在后面的步驟中完成)。
選擇實(shí)現(xiàn)一個(gè)簡(jiǎn)單的 15 抽頭低通濾波器 FIR,采樣率為 1Ms/s,通帶頻率為 200kHz,阻帶頻率為 355kHz,得到以下系數(shù):
-0.0265 0 0.0441 0 -0.0934 0 0.3139 0.5000 0.3139 0 -0.0934 0 0.0441 0 -0.0265
為 FIR 模塊創(chuàng)建設(shè)計(jì)文件
在 Vivado 項(xiàng)目中添加源文件。
在確定 FIR 的順序(抽頭數(shù))并獲得系數(shù)值后,接下來需要定義的下一組參數(shù)就是輸入樣本、輸出樣本和系數(shù)本身的位寬。
對(duì)于這個(gè) FIR,選擇將輸入樣本和系數(shù)寄存器設(shè)置為 16 位寬,并將輸出樣本寄存器設(shè)置為 32 位,因?yàn)閮蓚€(gè) 16 位值的乘積是一個(gè) 32 位值(兩個(gè)值的寬度相乘得到乘積的寬度,所以如果選擇了 8 位抽頭的 16 位輸入樣本,那么輸出樣本將為 24 位寬)。
這些值也都是帶符號(hào)的,因此 MSB 用作符號(hào)位,在選擇輸入樣本寄存器的初始寬度時(shí)一定要記住這一點(diǎn)。要在 Verilog 中將這些值設(shè)置為有符號(hào)數(shù)據(jù)類型,使用關(guān)鍵字signed :
regsigned[15:0]register_name;
接下來要解決的是如何在 Verilog 中處理系數(shù)值,小數(shù)點(diǎn)值需要轉(zhuǎn)換為定點(diǎn)值。由于所有系數(shù)值都小于 1,因此寄存器的所有 15 位(總共 16 位,MSB 是有符號(hào)位)都可以用于小數(shù)位。通常,必須決定要將寄存器中的多少位用于數(shù)字的整數(shù)部分與數(shù)字的小數(shù)部分。因此,轉(zhuǎn)換分?jǐn)?shù)值抽頭的數(shù)學(xué)是:(fractional coefficient value)*(2^(15))該乘積的小數(shù)值被四舍五入,并且如果系數(shù)為負(fù),則計(jì)算該值的二進(jìn)制補(bǔ)碼:
tap0=twos(-0.0265*32768)=0xFC9C tap1=0 tap2=0.0441*32768=1445.0688=1445=0x05A5 tap3=0 tap4=twos(-0.0934*32768)=0xF40C tap5=0 tap6=0.3139*32768=10285.8752=10285=0x282D tap7=0.5000*32768=16384=0x4000 tap8=0.3139*32768=10285.8752=10285=0x282D tap9=0 tap10=twos(-0.0934*32768)=0xF40C tap11=0 tap12=0.0441*32768=1445.0688=1445=0x05A5 tap13=0 tap14=twos(-0.0265*32768)=0xFC9C
現(xiàn)在我們終于準(zhǔn)備好關(guān)注 FIR 模塊的邏輯,第一個(gè)是循環(huán)緩沖區(qū),它引入串行輸入樣本流并為濾波器的 15 個(gè)抽頭創(chuàng)建一個(gè)包含 15 個(gè)輸入樣本的數(shù)組。
always@(posedgeclk) begin if(enable_buff==1'b1) begin buff0<=?in_sample; ????????????????????buff1?<=?buff0;???????? ????????????????????buff2?<=?buff1;????????? ????????????????????buff3?<=?buff2;?????? ????????????????????buff4?<=?buff3;?????? ????????????????????buff5?<=?buff4;??????? ????????????????????buff6?<=?buff5;???? ????????????????????buff7?<=?buff6;??????? ????????????????????buff8?<=?buff7;??????? ????????????????????buff9?<=?buff8;??????? ????????????????????buff10?<=?buff9;???????? ????????????????????buff11?<=?buff10;??????? ????????????????????buff12?<=?buff11;??????? ????????????????????buff13?<=?buff12;??????? ????????????????????buff14?<=?buff13;???? ????????????????end ????????end
接下來,乘法階段將每個(gè)樣本乘以每個(gè)系數(shù)值:
/*MultiplystageofFIR*/ always@(posedgeclk) begin if(enable_fir==1'b1) begin acc0<=?tap0?*?buff0; ????????????????????acc1?<=?tap1?*?buff1; ????????????????????acc2?<=?tap2?*?buff2; ????????????????????acc3?<=?tap3?*?buff3; ????????????????????acc4?<=?tap4?*?buff4; ????????????????????acc5?<=?tap5?*?buff5; ????????????????????acc6?<=?tap6?*?buff6; ????????????????????acc7?<=?tap7?*?buff7; ????????????????????acc8?<=?tap8?*?buff8; ????????????????????acc9?<=?tap9?*?buff9; ????????????????????acc10?<=?tap10?*?buff10; ????????????????????acc11?<=?tap11?*?buff11; ????????????????????acc12?<=?tap12?*?buff12; ????????????????????acc13?<=?tap13?*?buff13; ????????????????????acc14?<=?tap14?*?buff14; ????????????????end ????????end
乘法階段的結(jié)果值通過加法累加到寄存器中,最終成為濾波器的輸出數(shù)據(jù)流。
/*AccumulatestageofFIR*/ always@(posedgeclk) begin if(enable_fir==1'b1) begin m_axis_fir_tdata<=?acc0?+?acc1?+?acc2?+?acc3?+?acc4?+?acc5?+?acc6?+?acc7?+?acc8?+?acc9?+?acc10?+?acc11?+?acc12?+?acc13?+?acc14; ????????????????end ????????end
最后,邏輯的最后一部分是將數(shù)據(jù)流進(jìn)和流出 FIR 模塊的接口。AXI Stream 接口是最常見的接口之一。關(guān)鍵方面是允許控制上游和下游設(shè)備之間的數(shù)據(jù)流的tready和tvalid信號(hào)。這意味著 FIR 模塊需要向其下游設(shè)備提供tvalid信號(hào)以指示其輸出是有效數(shù)據(jù),并且如果下游設(shè)備解除其tready信號(hào),則能夠暫停(但仍保留)其輸出。FIR 模塊還必須能夠與其主端接口上的上游設(shè)備以同樣的方式運(yùn)行。
以下是 FIR 模塊的邏輯設(shè)計(jì)概述:
請(qǐng)注意tready和tvalid信號(hào)如何設(shè)置輸入循環(huán)緩沖器的使能值和 FIR 的乘法級(jí)以及數(shù)據(jù)或系數(shù)通過的每個(gè)寄存器都被聲明為有符號(hào)的。
FIR模塊Verilog代碼:
`timescale1ns/1ps moduleFIR( inputclk, inputreset, inputsigned[15:0]s_axis_fir_tdata, input[3:0]s_axis_fir_tkeep, inputs_axis_fir_tlast, inputs_axis_fir_tvalid, inputm_axis_fir_tready, outputregm_axis_fir_tvalid, outputregs_axis_fir_tready, outputregm_axis_fir_tlast, outputreg[3:0]m_axis_fir_tkeep, outputregsigned[31:0]m_axis_fir_tdata ); always@(posedgeclk) begin m_axis_fir_tkeep<=?4'hf; ????????end ???????? ????always?@?(posedge?clk) ????????begin ????????????if?(s_axis_fir_tlast?==?1'b1) ????????????????begin ????????????????????m_axis_fir_tlast?<=?1'b1; ????????????????end ????????????else ????????????????begin ????????????????????m_axis_fir_tlast?<=?1'b0; ????????????????end ????????end ???? ????//?15-tap?FIR? ????reg?enable_fir,?enable_buff; ????reg?[3:0]?buff_cnt; ????reg?signed?[15:0]?in_sample;? ????reg?signed?[15:0]?buff0,?buff1,?buff2,?buff3,?buff4,?buff5,?buff6,?buff7,?buff8,?buff9,?buff10,?buff11,?buff12,?buff13,?buff14;? ????wire?signed?[15:0]?tap0,?tap1,?tap2,?tap3,?tap4,?tap5,?tap6,?tap7,?tap8,?tap9,?tap10,?tap11,?tap12,?tap13,?tap14;? ????reg?signed?[31:0]?acc0,?acc1,?acc2,?acc3,?acc4,?acc5,?acc6,?acc7,?acc8,?acc9,?acc10,?acc11,?acc12,?acc13,?acc14;? ???? ????/*?Taps?for?LPF?running?@?1MSps?with?a?cutoff?freq?of?400kHz*/ ????assign?tap0?=?16'hFC9C;??//?twos(-0.0265?*?32768)?=?0xFC9C ????assign?tap1?=?16'h0000;??//?0 ????assign?tap2?=?16'h05A5;??//?0.0441?*?32768?=?1445.0688?=?1445?=?0x05A5 ????assign?tap3?=?16'h0000;??//?0 ????assign?tap4?=?16'hF40C;??//?twos(-0.0934?*?32768)?=?0xF40C ????assign?tap5?=?16'h0000;??//?0 ????assign?tap6?=?16'h282D;??//?0.3139?*?32768?=?10285.8752?=?10285?=?0x282D ????assign?tap7?=?16'h4000;??//?0.5000?*?32768?=?16384?=?0x4000 ????assign?tap8?=?16'h282D;??//?0.3139?*?32768?=?10285.8752?=?10285?=?0x282D ????assign?tap9?=?16'h0000;??//?0 ????assign?tap10?=?16'hF40C;?//?twos(-0.0934?*?32768)?=?0xF40C ????assign?tap11?=?16'h0000;?//?0 ????assign?tap12?=?16'h05A5;?//?0.0441?*?32768?=?1445.0688?=?1445?=?0x05A5 ????assign?tap13?=?16'h0000;?//?0 ????assign?tap14?=?16'hFC9C;?//?twos(-0.0265?*?32768)?=?0xFC9C ???? ????/*?This?loop?sets?the?tvalid?flag?on?the?output?of?the?FIR?high?once? ?????*?the?circular?buffer?has?been?filled?with?input?samples?for?the? ?????*?first?time?after?a?reset?condition.?*/ ????always?@?(posedge?clk?or?negedge?reset) ????????begin ????????????if?(reset?==?1'b0)?//if?(reset?==?1'b0?||?tvalid_in?==?1'b0) ????????????????begin ????????????????????buff_cnt?<=?4'd0; ????????????????????enable_fir?<=?1'b0; ????????????????????in_sample?<=?8'd0; ????????????????end ????????????else?if?(m_axis_fir_tready?==?1'b0?||?s_axis_fir_tvalid?==?1'b0) ????????????????begin ????????????????????enable_fir?<=?1'b0; ????????????????????buff_cnt?<=?4'd15; ????????????????????in_sample?<=?in_sample; ????????????????end ????????????else?if?(buff_cnt?==?4'd15) ????????????????begin ????????????????????buff_cnt?<=?4'd0; ????????????????????enable_fir?<=?1'b1; ????????????????????in_sample?<=?s_axis_fir_tdata; ????????????????end ????????????else ????????????????begin ????????????????????buff_cnt?<=?buff_cnt?+?1; ????????????????????in_sample?<=?s_axis_fir_tdata; ????????????????end ????????end??? ????always?@?(posedge?clk) ????????begin ????????????if(reset?==?1'b0?||?m_axis_fir_tready?==?1'b0?||?s_axis_fir_tvalid?==?1'b0) ????????????????begin ????????????????????s_axis_fir_tready?<=?1'b0; ????????????????????m_axis_fir_tvalid?<=?1'b0; ????????????????????enable_buff?<=?1'b0; ????????????????end ????????????else ????????????????begin ????????????????????s_axis_fir_tready?<=?1'b1; ????????????????????m_axis_fir_tvalid?<=?1'b1; ????????????????????enable_buff?<=?1'b1; ????????????????end ????????end ???? ????/*?Circular?buffer?bring?in?a?serial?input?sample?stream?that? ?????*?creates?an?array?of?15?input?samples?for?the?15?taps?of?the?filter.?*/ ????always?@?(posedge?clk) ????????begin ????????????if(enable_buff?==?1'b1) ????????????????begin ????????????????????buff0?<=?in_sample; ????????????????????buff1?<=?buff0;???????? ????????????????????buff2?<=?buff1;????????? ????????????????????buff3?<=?buff2;?????? ????????????????????buff4?<=?buff3;?????? ????????????????????buff5?<=?buff4;??????? ????????????????????buff6?<=?buff5;???? ????????????????????buff7?<=?buff6;??????? ????????????????????buff8?<=?buff7;??????? ????????????????????buff9?<=?buff8;??????? ????????????????????buff10?<=?buff9;???????? ????????????????????buff11?<=?buff10;??????? ????????????????????buff12?<=?buff11;??????? ????????????????????buff13?<=?buff12;??????? ????????????????????buff14?<=?buff13;???? ????????????????end ????????????else ????????????????begin ????????????????????buff0?<=?buff0; ????????????????????buff1?<=?buff1;???????? ????????????????????buff2?<=?buff2;????????? ????????????????????buff3?<=?buff3;?????? ????????????????????buff4?<=?buff4;?????? ????????????????????buff5?<=?buff5;??????? ????????????????????buff6?<=?buff6;???? ????????????????????buff7?<=?buff7;??????? ????????????????????buff8?<=?buff8;??????? ????????????????????buff9?<=?buff9;??????? ????????????????????buff10?<=?buff10;???????? ????????????????????buff11?<=?buff11;??????? ????????????????????buff12?<=?buff12;??????? ????????????????????buff13?<=?buff13;??????? ????????????????????buff14?<=?buff14; ????????????????end ????????end ???????? ????/*?Multiply?stage?of?FIR?*/ ????always?@?(posedge?clk) ????????begin ????????????if?(enable_fir?==?1'b1) ????????????????begin ????????????????????acc0?<=?tap0?*?buff0; ????????????????????acc1?<=?tap1?*?buff1; ????????????????????acc2?<=?tap2?*?buff2; ????????????????????acc3?<=?tap3?*?buff3; ????????????????????acc4?<=?tap4?*?buff4; ????????????????????acc5?<=?tap5?*?buff5; ????????????????????acc6?<=?tap6?*?buff6; ????????????????????acc7?<=?tap7?*?buff7; ????????????????????acc8?<=?tap8?*?buff8; ????????????????????acc9?<=?tap9?*?buff9; ????????????????????acc10?<=?tap10?*?buff10; ????????????????????acc11?<=?tap11?*?buff11; ????????????????????acc12?<=?tap12?*?buff12; ????????????????????acc13?<=?tap13?*?buff13; ????????????????????acc14?<=?tap14?*?buff14; ????????????????end ????????end???? ???????? ?????/*?Accumulate?stage?of?FIR?*/??? ????always?@?(posedge?clk)? ????????begin ????????????if?(enable_fir?==?1'b1) ????????????????begin ????????????????????m_axis_fir_tdata?<=?acc0?+?acc1?+?acc2?+?acc3?+?acc4?+?acc5?+?acc6?+?acc7?+?acc8?+?acc9?+?acc10?+?acc11?+?acc12?+?acc13?+?acc14; ????????????????end ????????end????? ???? ???? endmodule
創(chuàng)建仿真文件
要測(cè)試 FIR 模塊,需要?jiǎng)?chuàng)建一個(gè)測(cè)試平臺(tái)作為其仿真源:
在 FIR 模塊中有兩個(gè)主要的東西需要測(cè)試:濾波器算法和 AXI 流接口。為實(shí)現(xiàn)這一點(diǎn),測(cè)試臺(tái)中創(chuàng)建了一個(gè)狀態(tài)機(jī),它生成一個(gè)簡(jiǎn)單的 200kHz 正弦波,并切換從屬端的有效信號(hào)和 FIR 接口主控端的tready信號(hào)。
FIR 模塊的測(cè)試平臺(tái):
`timescale1ns/1ps moduletb_FIR; regclk,reset,s_axis_fir_tvalid,m_axis_fir_tready; regsigned[15:0]s_axis_fir_tdata; wirem_axis_fir_tvalid; wire[3:0]m_axis_fir_tkeep; wire[31:0]m_axis_fir_tdata; /* *100Mhz(10ns)clock */ alwaysbegin clk=1;#5; clk=0;#5; end alwaysbegin reset=1;#20; reset=0;#50; reset=1;#1000000; end alwaysbegin s_axis_fir_tvalid=0;#100; s_axis_fir_tvalid=1;#1000; s_axis_fir_tvalid=0;#50; s_axis_fir_tvalid=1;#998920; end alwaysbegin m_axis_fir_tready=1;#1500; m_axis_fir_tready=0;#100; m_axis_fir_tready=1;#998400; end /*InstantiateFIRmoduletotest.*/ FIRFIR_i( .clk(clk), .reset(reset), .s_axis_fir_tdata(s_axis_fir_tdata), .s_axis_fir_tkeep(s_axis_fir_tkeep), .s_axis_fir_tlast(s_axis_fir_tlast), .s_axis_fir_tvalid(s_axis_fir_tvalid), .m_axis_fir_tready(m_axis_fir_tready), .m_axis_fir_tvalid(m_axis_fir_tvalid), .s_axis_fir_tready(s_axis_fir_tready), .m_axis_fir_tlast(m_axis_fir_tlast), .m_axis_fir_tkeep(m_axis_fir_tkeep), .m_axis_fir_tdata(m_axis_fir_tdata)); reg[4:0]state_reg; reg[3:0]cntr; parameterwvfm_period=4'd4; parameterinit=5'd0; parametersendSample0=5'd1; parametersendSample1=5'd2; parametersendSample2=5'd3; parametersendSample3=5'd4; parametersendSample4=5'd5; parametersendSample5=5'd6; parametersendSample6=5'd7; parametersendSample7=5'd8; /*Thisstatemachinegeneratesa200kHzsinusoid.*/ always@(posedgeclkorposedgereset) begin if(reset==1'b0) begin cntr<=?4'd0; ????????????????????s_axis_fir_tdata?<=?16'd0; ????????????????????state_reg?<=?init; ????????????????end ????????????else ????????????????begin ????????????????????case?(state_reg) ????????????????????????init?:?//0 ????????????????????????????begin ????????????????????????????????cntr?<=?4'd0; ????????????????????????????????s_axis_fir_tdata?<=?16'h0000; ????????????????????????????????state_reg?<=?sendSample0; ????????????????????????????end ???????????????????????????? ????????????????????????sendSample0?:?//1 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'h0000; ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample1; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample0; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample1?:?//2 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'h5A7E;? ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample2; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample1; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample2?:?//3 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'h7FFF; ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample3; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample2; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample3?:?//4 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'h5A7E; ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample4; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample3; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample4?:?//5 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'h0000; ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample5; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample4; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample5?:?//6 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'hA582;? ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample6; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample5; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample6?:?//6 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'h8000;? ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample7; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample6; ????????????????????????????????????end ????????????????????????????end? ???????????????????????? ????????????????????????sendSample7?:?//6 ????????????????????????????begin ????????????????????????????????s_axis_fir_tdata?<=?16'hA582;? ???????????????????????????????? ????????????????????????????????if?(cntr?==?wvfm_period) ????????????????????????????????????begin ????????????????????????????????????????cntr?<=?4'd0; ????????????????????????????????????????state_reg?<=?sendSample0; ????????????????????????????????????end ????????????????????????????????else ????????????????????????????????????begin? ????????????????????????????????????????cntr?<=?cntr?+?1; ????????????????????????????????????????state_reg?<=?sendSample7; ????????????????????????????????????end ????????????????????????????end????????????????????? ???????????????????? ????????????????????endcase ????????????????end ????????end ???? endmodule
運(yùn)行行為仿真
FIR 模塊及其測(cè)試平臺(tái)文件就位后,從 Flow Navigator 窗口啟動(dòng) Vivado 中的仿真器,選擇 Run Behavioral Simulation 選項(xiàng)。
如行為仿真所示,F(xiàn)IR 正確過濾信號(hào)并正確響應(yīng) AXI 流信號(hào)。
總結(jié)
代碼都在上面大家有興趣可以自行運(yùn)行,但是大家可能會(huì)注意到,這個(gè) FIR 模塊在設(shè)計(jì)上運(yùn)行綜合和實(shí)現(xiàn)時(shí)時(shí)序應(yīng)該是不能通過的。我們將在下一篇文章中詳細(xì)介紹如何在無(wú)法滿足時(shí)序要求時(shí)重新設(shè)計(jì)你的設(shè)計(jì)~
審核編輯:湯梓紅
-
FPGA
+關(guān)注
關(guān)注
1643文章
21923瀏覽量
612353 -
濾波器
+關(guān)注
關(guān)注
162文章
8012瀏覽量
180561 -
數(shù)字信號(hào)處理
+關(guān)注
關(guān)注
16文章
567瀏覽量
46446 -
FIR
+關(guān)注
關(guān)注
4文章
151瀏覽量
33899 -
Verilog
+關(guān)注
關(guān)注
28文章
1364瀏覽量
111538
原文標(biāo)題:FPGA 的數(shù)字信號(hào)處理:Verilog 實(shí)現(xiàn)簡(jiǎn)單的 FIR 濾波器
文章出處:【微信號(hào):Open_FPGA,微信公眾號(hào):OpenFPGA】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
DSP in FPGA:FIR濾波器(一)
DSP in FPGA:FIR濾波器(二)
matlab與FPGA數(shù)字信號(hào)處理系列 Verilog 實(shí)現(xiàn)并行 FIR 濾波器
進(jìn)群免費(fèi)領(lǐng)FPGA學(xué)習(xí)資料!數(shù)字信號(hào)處理、傅里葉變換與FPGA開發(fā)等
【參考書籍】基于FPGA的數(shù)字信號(hào)處理——高亞軍著
fpga實(shí)現(xiàn)濾波器
并行FIR濾波器Verilog設(shè)計(jì)
數(shù)字信號(hào)處理的FPGA實(shí)現(xiàn)
什么是fir數(shù)字濾波器 什么叫FIR濾波器
數(shù)字信號(hào)處理的FPGA實(shí)現(xiàn)
如何使用FPGA實(shí)現(xiàn)FIR抽取濾波器的設(shè)計(jì)

FIR濾波器代碼及仿真設(shè)計(jì)

FPGA的數(shù)字信號(hào)處理:重寫FIR邏輯以滿足時(shí)序要求

基于FPGA實(shí)現(xiàn)FIR數(shù)字濾波器

評(píng)論