FPGA高速ADC接口實戰——250MSPS採樣率ADC9481

1、前言緩存

  最近忙於碩士畢業設計和論文,沒有太多時間編寫博客,現總結下以前在某個項目中用到的一個高速ADC接口設計部分。ADC這一器件常常用於無線通訊、傳感、測試測量等領域。目前數字系統對高速數據採集的需求與日俱增,本文使用了米聯客的一款速率較高的AD/DA模塊ADQ9481來闡述利用FPGA設計高速ADC接口的技術要點。架構

2、ADC硬件特性分析異步

  首先必須經過datasheet分析其核心參數、接口定義和時序要求。ADC9481的採樣率爲250MSPS,精度8bit。其原理結構圖以下:async

  主要引腳說明:ide

  CLK+-:差分時鐘輸入,信號頻率爲250MHz測試

  VIN+-:模擬信號輸入,範圍是1Vppspa

  VREF:電壓參考輸入/輸出,這裏使用內部固定參考電壓模式debug

  SENSE:參考模式選擇設計

  D7A~D0A:通道A數字信號輸出3d

   D7B~D0B:通道B數字信號輸出

  DCO+-:數字差分時鐘輸出,信號頻率爲125MHz

   S1:數據格式選擇,該接口電壓決定數格式時原碼仍是補碼

  PDWN:低功耗選通

  接下來看看接口時序:

  很容易看出A和B兩個數字輸出通道是交替輸出的,通道A在DCO+上升沿輸出,B在DCO-上升沿輸出。DCO+-的頻率僅是採樣率250MHz的一半,也就是下降了對數字系統處理速率的要求。

3、ADC接口設計

   根據上述時序關係可知,FPGA端須要在DCO+上升沿採集通道B數據,在DCO-上升沿採集通道A數據。而且因爲在DCO+-同一變化沿時刻,通道A爲前一個數據,所以要注意數據的採集順序。這類數據採集的廣泛作法是將數據存入到RAM中,而後利用本地時鐘同步。具體方法是:按照兩通道的數據順序對數據進行拼接,以後緩存到異步FIFO中。本地PLL生成的125MHz時鐘做爲讀側和後續處理時鐘信號。這裏就要利用Xilinx FPGA的「原語」中的IBUFDS+BUFG,依次是差分輸入緩衝器和全局緩衝器。前者可將差分信號轉變爲單端信號,後者則可以讓時鐘信號到達FPGA內部邏輯引腳的時延和抖動最小。綜上,ADC接口硬件架構如圖:

 4、HDL代碼編寫

   根據前文所述的硬件架構,ADC接口HDL代碼以下:

  1 `timescale 1ns / 1ps
  2 
  3 module adc_interface#(parameter WIDTH = 8,
  4                                 FRAME_LEN = 512
  5                                 //WAIT_CYC = 125_000_000//1s = 1000_000_000ns   1000_000_000/8 = 125_000_000
  6               )
  7     (
  8     input                       clk0,     //125MHZ
  9     input                       clk1,
 10     
 11     input       [WIDTH-1:0]     da,
 12     input       [WIDTH-1:0]     db,
 13     output                      adc_pd,//省電模式選擇
 14 
 15     output                      pll_ce,
 16     output                      pll_rst_n,
 17     output                      pen,
 18     
 19     input                       user_clk,//125MHZ
 20     input                       rst_n,
 21     input                       en,
 22     output reg [WIDTH*2-1:0]    dout = 0,
 23     output reg                  dout_vld = 0
 24     );
 25 
 26     function integer clogb2 (input integer bit_depth);
 27       begin
 28         for(clogb2=0; bit_depth>0; clogb2=clogb2+1)
 29           bit_depth = bit_depth >> 1;
 30       end
 31     endfunction
 32 
 33     localparam DATA_CNT_W = clogb2(FRAME_LEN-1);
 34 
 35 (*DONT_TOUCH = "true"*)reg setup_flag = 0;
 36 reg [WIDTH-1:0] data_a = 0,data_b = 0;
 37 reg data_a_vld = 0,data_b_vld = 0;
 38 reg wr_en = 0;
 39 reg [WIDTH*2-1:0] wr_data = 0;
 40 reg rd_en = 0;
 41 wire empty;
 42 wire full;
 43 wire [WIDTH*2-1:0] rd_data;
 44 (*DONT_TOUCH = "true"*)wire en_pos;
 45 (*DONT_TOUCH = "true"*)reg [ (DATA_CNT_W-1):0]  data_cnt  =0   ;
 46 wire        add_data_cnt ;
 47 wire        end_data_cnt ;
 48 reg en_r0 = 0,en_r1 = 0,en_r2 = 0,en_r3 = 0;
 49 
 50     assign pll_ce     = 1'b1;
 51     assign pll_rst_n  = 1'b1;
 52     assign adc_pd     = 1'b0;
 53     assign pen        = 1'b1;
 54 
 55     /***************************採集觸發**************************************/
 56     //異步處理
 57     always@(posedge clk0)begin
 58         en_r0 <= en;
 59         en_r1 <= en_r0;
 60         en_r2 <= en_r1;
 61         en_r3 <= en_r2;
 62     end
 63 
 64     assign en_pos = en_r2 == 1'b1 && en_r3 == 1'b0;
 65 
 66 always  @(posedge clk0 or negedge rst_n)begin
 67     if(rst_n==1'b0)begin
 68         setup_flag <= 0;
 69     end
 70     else if(end_data_cnt)
 71         setup_flag <= 0;
 72     else if(en_pos)begin
 73         setup_flag <= 1'b1;
 74     end
 75 end
 76 
 77 always @(posedge clk0 or negedge rst_n) begin 
 78     if (rst_n==0) begin
 79         data_cnt <= 0; 
 80     end
 81     else if(add_data_cnt) begin
 82         if(end_data_cnt)
 83             data_cnt <= 0; 
 84         else
 85             data_cnt <= data_cnt+1 ;
 86    end
 87 end
 88 assign add_data_cnt = (setup_flag);
 89 assign end_data_cnt = add_data_cnt  && data_cnt == (FRAME_LEN)-1 ;
 90 
 91 
 92 /***************************clk0(dco_p)採集DB**************************************/
 93     always@(posedge clk0 or negedge rst_n)begin
 94         if(rst_n == 0)
 95             data_b <= 0;
 96         else
 97             data_b <= db;
 98     end
 99 
100     always@(posedge clk0 or negedge rst_n)begin
101         if(rst_n == 0)
102             data_b_vld <= 0;
103         else if(setup_flag)
104             data_b_vld <= 1'b1;
105         else
106             data_b_vld <= 0;
107     end
108 
109  /****************************clk1(dco_n)採集DA**************************************/
110     always  @(negedge clk0 or negedge rst_n)begin
111         if(rst_n==1'b0)begin
112             data_a <= 0;
113         end
114         else begin
115             data_a <= da;
116         end
117     end
118 
119     always@(negedge clk0 or negedge rst_n)begin
120         if(rst_n == 0)begin
121             data_a_vld <= 0;
122         end
123         else if(setup_flag)begin
124             data_a_vld <= 1'b1;
125         end
126         else
127             data_a_vld <= 0;
128     end
129  /****************************FIFO寫邏輯**************************************/
130 //FIFO:width 16bit  depth 16 async
131 
132 always  @(negedge clk0 or negedge rst_n)begin
133     if(rst_n==1'b0)begin
134         wr_en <= 0;
135     end
136     else if(data_a_vld & data_b_vld)begin
137         wr_en <= 1'b1;
138     end
139     else
140         wr_en <= 0;
141 end
142 
143 always  @(negedge clk0 or negedge rst_n)begin
144     if(rst_n==1'b0)begin
145         wr_data <= 0;
146     end
147     else begin
148         wr_data <= {data_b,data_a};//高字節爲後一個數據
149     end
150 end
151 
152     /****************************FIFO讀側邏輯**************************************/
153     //非空即讀
154     always@(*)begin
155         if(~empty)
156             rd_en = 1'b1;
157         else
158             rd_en = 0;
159     end
160 
161     always  @(posedge user_clk or negedge rst_n)begin
162         if(rst_n == 0)
163             dout <= 0;
164         else
165             dout <= rd_data;
166     end
167 
168     always  @(posedge user_clk or negedge rst_n)begin
169         if(rst_n == 0)
170             dout_vld <= 0;
171         else if(rd_en)begin
172             dout_vld <= 1'b1;
173         end
174         else
175             dout_vld <= 0;
176     end
177 
178 
179 //FIFO instance 
180 fifo_generator_2 interface_fifo (
181   .wr_clk(~clk0),                // input wire wr_clk
182   .rd_clk(user_clk),                // input wire rd_clk
183   .din(wr_data),                      // input wire [15 : 0] din
184   .wr_en(wr_en),                  // input wire wr_en
185   .rd_en(rd_en),                  // input wire rd_en
186   .dout(rd_data),                    // output wire [15 : 0] dout
187   .full(full),                    // output wire full
188   .empty(empty),                  // output wire empty
189   .rd_data_count(rd_data_count),  // output wire [3 : 0] rd_data_count
190   .wr_data_count(wr_data_count)  // output wire [3 : 0] wr_data_count
191 );
192 
193 
194 endmodule
adc_interface

  頂層模塊代碼:

  1 `timescale 1ns / 1ps
  2 
  3 module top#(parameter DATA_W = 8,//改動參數須要從新配置IP核
  4                       CHANNEL_NUM = 2)
  5    (
  6     input                                       dco_p,//125MHZ 
  7     input                                       dco_n,
  8     input       [DATA_W-1:0]                    adc_p1,//通道A
  9     input       [DATA_W-1:0]                    adc_p2,//通道B
 10     output                                      adc_pd,
 11     output                                      pll_ce,
 12     output                                      pll_rst_n,
 13     output                                      pen,
 14     
 15     //user interface signals
 16     
 17     input                                       clk,//100M
 18     input                                       rst_n,
 19     input                                       en//上升沿有效 有效一次則將以後採集到的一幀數據寫入到FFT模塊進行運算
 20     );
 21 /*********************************parameters*******************************************/
 22     function integer clogb2 (input integer bit_depth);
 23       begin
 24         for(clogb2=0; bit_depth>0; clogb2=clogb2+1)
 25           bit_depth = bit_depth >> 1;
 26       end
 27     endfunction
 28 
 29     localparam  FFT_W = 20,
 30                 FFT_N = 1024,
 31                 DATA_EACH_CHANNEL = FFT_N/CHANNEL_NUM;
 32 
 33     //log2 
 34     localparam DECH_W = clogb2(DATA_EACH_CHANNEL-1);
 35 
 36     
 37 /*********************************variables*******************************************/
 38 genvar ii;
 39 wire clk_out0,locked0;
 40 reg locked0_r0 = 0,locked0_r1 = 0;
 41 wire clk_user;
 42 wire dco;
 43 wire dco_bufg;
 44 
 45 wire [DATA_W*2-1:0] data_adc;
 46 wire data_adc_vld;
 47 (*DONT_TOUCH = "true"*)wire [DATA_W*2-1:0] din;
 48 (*DONT_TOUCH = "true"*)wire din_vld;
 49 wire din_sop,din_eop;
 50 reg [ (DECH_W-1):0]  data_cnt =0    ;
 51 wire        add_data_cnt ;
 52 wire        end_data_cnt ;
 53 
 54 /******************************clock generators****************************************/
 55    
 56 //user clock generator
 57  clk_wiz_0 user_clock_gen
 58    (
 59     // Clock out ports
 60     .clk_out1(clk_out0),     // output clk_out0  125MHZ
 61     // Status and control signals
 62     .locked(locked0),       // output locked
 63    // Clock in ports
 64     .clk_in1(clk));      // input clk_in1 100MHZ 
 65 
 66     //pll lock信號同步
 67     always@(posedge clk_out0)begin
 68         locked0_r0 <= locked0;
 69         locked0_r1 <= locked0_r0;
 70     end
 71 
 72     assign clk_user = clk_out0 & locked0_r1;
 73   
 74 
 75 // ADC clock generator
 76 IBUFDS #(
 77     .DIFF_TERM("FALSE"),
 78     .IBUF_LOW_PWR("FALSE"),
 79     .IOSTANDARD("DEFAULT")
 80 ) IBUFDS_inst (
 81     .O(dco),
 82     .I(dco_p),
 83     .IB(dco_n)
 84 );
 85 
 86 BUFG BUFG_inst(
 87     .O(dco_bufg),
 88     .I(dco)
 89 );
 90    
 91 /**********************************ADC interface module***********************************/
 92 adc_interface#(.WIDTH(DATA_W),
 93                .FRAME_LEN(DATA_EACH_CHANNEL))
 94 u_adc_interface
 95 (
 96     //adc -> fpga
 97     . clk0        (dco_bufg)  ,     //125MHZ 與dco_p同相
 98     . clk1        (~dco_bufg),      //125MHZ 與dco_n反相
 99     . da          (adc_p1)  ,
100     . db          (adc_p2)  ,
101     //fpga -> adc
102     . adc_pd      (adc_pd)  ,//省點模式選擇
103     . pll_ce      (pll_ce)  ,
104     . pll_rst_n   (pll_rst_n)  ,
105     . pen         (pen),
106     //user
107     . user_clk    (clk_user),//125MHZ
108     . rst_n       (rst_n),
109     . dout        (data_adc),//debug
110     . dout_vld    (data_adc_vld),
111     . en          (en)//上升沿有效
112     );
113 //data counter
114 always @(posedge clk_user or negedge rst_n) begin 
115     if (rst_n==0) begin
116         data_cnt <= 0; 
117     end
118     else if(add_data_cnt) begin
119         if(end_data_cnt)
120             data_cnt <= 0; 
121         else
122             data_cnt <= data_cnt+1 ;
123    end
124 end
125 
126 assign add_data_cnt = data_adc_vld ;
127 assign end_data_cnt = add_data_cnt  && data_cnt == (DATA_EACH_CHANNEL)-1 ;
128 
129 //input data to the user defined logic
130     assign din_sop  = add_data_cnt && data_cnt == 0;
131     assign din_eop  = end_data_cnt;  
132     assign din      = data_adc;
133     assign din_vld  = data_adc_vld;
134 
135 
136 //user logic end  
137 
138 endmodule
top

  上述代碼是以前作ADC採集信號頻譜分析的部分代碼,所以adc_interface模塊中每觸發一次則連續採集一幀數據長度,用於FFT運算。用戶能夠根據項目需求自行改動。頂層模塊中則例化IBUFDS+BUFG原語,以及後續的自定義處理模塊。

5、板級調試

  行爲仿真是FPGA開發中必不可少的重要環節,經過充分測試可節省不少調試時間,這裏僅給出板級調試結果。信號發生器產生三角波,利用ILA抓取芯片內部實時數據,並以模擬形式顯示:

  因爲在接口模塊中將兩通道輸入拼接爲一個數據,這裏拆分後觀察數據數值。可見拼接後數據波形呈現三角波形狀,且幅值增大過程當中高字節較大,幅值下降過程當中高字節較小,說明數據拼接順序無誤,高字節爲當前節拍後一個採樣數據。兩路輸出數據最高位爲0,證實輸出數據格式是天然二進制數。

  若想細緻地觀察數據的模擬形狀,能夠經過靈活的TCL腳本將ILA抓取數據導出,並在MATLAB中查看。TCL命令爲:

write_hw_ila_data E:/fpga_files/wave_file.csv [upload_hw_ila_data hw_ila_1] -csv_file –forcewave

  命令格式是:write_hw_ila_data <文件路徑及文件名> [upload_hw_ila_data <ILA名稱>] -csv_file -forcewave。鍵入該命令後,指定路徑下會產生CSV文件。如讓信號發生器產生頻率爲1MHz,峯峯值是1Vpp,偏移幅值是0.7V的正弦波。導出ILA抓取數據,並在MATLAB中繪製曲線如圖:

   總體來看仍是比較簡單的。後邊若有機會接觸採樣率更高的ADC芯片,會總結基於Select I/O IP Core的LVDS接口設計。

相關文章
相關標籤/搜索