首页 > 其他分享 >11 ADC模块FEP-DAQ422X采集显示波形方案

11 ADC模块FEP-DAQ422X采集显示波形方案

时间:2023-12-27 19:46:48浏览次数:39  
标签:11 reset vtc clk ads422x cnt b0 ADC FEP

软件版本:VIVADO2021.1

操作系统:WIN10 64bit

硬件平台:适用XILINX A7/K7/Z7/ZU/KU系列FPGA

登录米联客(MiLianKe)FPGA社区-www.uisrc.com观看免费视频课程、在线答疑解惑!

1概述

本方案通过把DAQ422X采集到的数据,通过前面已经完成的示波器显示驱动进行在屏幕上显示ADC采集的波形数据。

2系统框图

3产品介绍

FEP-DAQ003- 12- 125M-250M-2数据采集模块采用一颗 TI 的ADS4225/ ADS4229低功耗高性能模数转换芯片,实现了 2 通道125/250MSPS 模数转换,并且支持 2 路数字 IO 输入/输出触发功能。

通过设置不同的通信模式,ADS4225/ADS4229 具有 LVDS DDR 接口模式和 LVCMOS SDR 接口两种模式。

3.1 硬件参数概述

 

FEP-DAQ003- 12- 125M-2/FEP-DAQ003- 12-250M-2

ADC 芯片

AD4225/4229

采样精度

12bit

-3db 带宽

320M(前置运放带宽决定)

IO 电平

1.8V 或者 3.3V 可选

采样频率

ADS4225 0~125M

ADS4229 0~250M

模拟通道

2 个

触发 IO

2 个

输入电平

-5V~+5V

数据格式

二进制补码(~2048~+2047)

信噪比 170M(SNR)

71.4dBFS

无杂散动态范围 170M(SFDR)

88dBc

功耗 165M

332mW

占用 IO 数量

40 个 GPIO

3.2 引脚定义

3.2.1 SMA 引脚定义

引脚号

引脚名称

描述

PA

DAI

模拟输入通道 1

PB

DBI

模拟输入通道 2

TA

TriA

双向触发输入/输出 1

TB

TriB

双向触发输入/输出 2

3.2.2 ADS422X芯片功能引脚定义

3.2.2.1 LVDS 下接口定义

3.2.2.2 LVCMOS 下接口定义

 

LVDS 和CMOS 模式的数据输出时序

LVDS 的时序要求

CMOS 模式的时序要求

3.3 数据模式设置

设置 SEN 脚的工作电平,来设置 ADS4225/ADS4229 的工作模式

 

SEN 脚电平

模式

0V

二进制补码和并行 CMOS 输出

0.375AVDD

偏移二进制和并行 CMOS 输出

0.625AVDD

偏移二进制和 DDR LVDS 输出

4硬件电路分析

硬件接口和子卡模块请阅读"附录 1"

配套工程的 FPGA PIN 脚定义路径为 fpga_prj/uisrc/04_pin/ fpga_pin.xdc。

5程序源码

关于HDMI输出IP的部分这里不再介绍,VTC时序设计部分这里也不详细介绍。如果读者这些基础知识不清楚的,请阅读前面的实验。

5.1ADS422X采集驱动

ADS422X 数据采集通过 IDELAY 模块调整数据相对时钟的延迟,用来设置最佳采样时刻。

`timescale 1 ns / 1 ns

 

module uiads422X_parallel#

(

parameter  FAMILY    = "7SERIES",  

parameter  integer  DATA_WIDTH     = 12,

parameter  integer  DELAY_SETA     = 12,

parameter  integer  DELAY_SETB     = 12

)

(

input       I_refclk,

input       I_reset,

 

input       I_ad_clk,

input [DATA_WIDTH   -1 :0]I_ad_da,

input [DATA_WIDTH   -1 :0]I_ad_db,

output      [DATA_WIDTH   -1 :0]O_ad_da,

output      [DATA_WIDTH   -1 :0]O_ad_db,

output      O_ad_reset,

output      O_ad_sen,

output      O_ad_sclk,

output      O_ad_card_en

);

 

wire [DATA_WIDTH   -1:0] da_buf,db_buf,da_buf_delay,db_buf_delay;

wire      idelayctrl_reset_sync;

reg       idelayctrl_reset;

reg [3 :0]idelay_reset_cnt;

reg [15:0] cnt = 0;

 

assign    O_ad_card_en   = cnt[15];

assign    O_ad_reset = 1'b1;

assign    O_ad_sen   = 1'b0;

assign    O_ad_sclk  = 1'b0;

 

always @(posedge I_refclk)begin

    if(!cnt[15])

        cnt <= cnt +1'b1;

end

 

// Create a synchronous reset in the IDELAYCTRL refclk clock domain.

reset_sync idelayctrl_reset_gen

(

.clk              (I_refclk),

.enable           (1'b1),

.reset_in         (I_reset),

.reset_out        (idelayctrl_reset_sync)

);    

 

// The IDELAYCTRL must experience a pulse which is at least 50 ns in

// duration.  This is ten clock cycles of the 200MHz refclk.  Here we

// drive the reset pulse for 12 clock cycles.

always @(posedge I_refclk)begin

      if (idelayctrl_reset_sync == 1'b0) begin

         idelay_reset_cnt     <= 4'b0000;

         idelayctrl_reset     <= 1'b1;

      end

      else begin

         case (idelay_reset_cnt)

            4'b0000 : idelay_reset_cnt <= 4'b0001;

            4'b0001 : idelay_reset_cnt <= 4'b0010;

            4'b0010 : idelay_reset_cnt <= 4'b0011;

            4'b0011 : idelay_reset_cnt <= 4'b0100;

            4'b0100 : idelay_reset_cnt <= 4'b0101;

            4'b0101 : idelay_reset_cnt <= 4'b0110;

            4'b0110 : idelay_reset_cnt <= 4'b0111;

            4'b0111 : idelay_reset_cnt <= 4'b1000;

            4'b1000 : idelay_reset_cnt <= 4'b1001;

            4'b1001 : idelay_reset_cnt <= 4'b1010;

            4'b1010 : idelay_reset_cnt <= 4'b1011;

            4'b1011 : idelay_reset_cnt <= 4'b1100;

            default : idelay_reset_cnt <= 4'b1100;

         endcase

         if (idelay_reset_cnt == 4'b1100) begin

            idelayctrl_reset  <= 1'b0;

         end

         else begin

            idelayctrl_reset  <= 1'b1;

         end

      end

end

     

genvar i;

generate

for (i = 0 ; i < DATA_WIDTH ; i = i+1) begin : DAQ_DATA

IBUF #(

.IBUF_LOW_PWR("TRUE"),  // Low power (TRUE) vs. performance (FALSE) setting for referenced I/O standards

.IOSTANDARD("DEFAULT")  // Specify the input I/O standard

)IBUF_da (

.O(da_buf[i]),     // Buffer output

.I(I_ad_da[i])      // Buffer input (connect directly to top-level port)

);

     

 

IBUF #(

.IBUF_LOW_PWR("TRUE"),  // Low power (TRUE) vs. performance (FALSE) setting for referenced I/O standards

.IOSTANDARD("DEFAULT")  // Specify the input I/O standard

)IBUF_db (

.O(db_buf[i]),     // Buffer output

.I(I_ad_db[i])      // Buffer input (connect directly to top-level port)

);

     

end

endgenerate

 

generate  if(FAMILY == "ULTRASCALE" || FAMILY == "ULTRASCALE_PLUS")begin : ULTRASCALE_FAMILY

 

IDELAYCTRL  #(

.SIM_DEVICE ("ULTRASCALE")

)

idelayctrl_inst(

.RDY       (idelayctrl_ready),

.REFCLK    (I_refclk),

.RST       (idelayctrl_reset)

);

 

    genvar k;

     for (k=0; k<DATA_WIDTH ; k=k+1)

    begin : rxdata_in_bus

    IDELAYE3 #(

    .DELAY_VALUE      (DELAY_SETA),

    .DELAY_TYPE       ("FIXED"),

    .REFCLK_FREQUENCY (300.000),

    .SIM_DEVICE       (FAMILY)

    )

    IDELAYE_A (

    .IDATAIN          (da_buf[k]),

    .DATAOUT          (da_buf_delay[k]),

    .DATAIN           (1'b0),

    .CLK              (1'b0),

    .CE               (1'b0),

    .INC              (1'b0),

    .CNTVALUEIN       (9'h0),

    .CNTVALUEOUT      (),

    .LOAD             (1'b0),

    .RST              (1'b0),

    .CASC_IN          (1'b0),

    .CASC_RETURN      (1'b0),

    .CASC_OUT         (),

    .EN_VTC           (1'b1)

    );

    IDELAYE3 #(

    .DELAY_VALUE      (DELAY_SETB),

    .DELAY_TYPE       ("FIXED"),

    .REFCLK_FREQUENCY (300.000),

    .SIM_DEVICE       (FAMILY)  

    )

    IDELAYE_B (

    .IDATAIN          (db_buf[k]),

    .DATAOUT          (db_buf_delay[k]),

    .DATAIN           (1'b0),

    .CLK              (1'b0),

    .CE               (1'b0),

    .INC              (1'b0),

    .CNTVALUEIN       (9'h0),

    .CNTVALUEOUT      (),

    .LOAD             (1'b0),

    .RST              (1'b0),

    .CASC_IN          (1'b0),

    .CASC_RETURN      (1'b0),

    .CASC_OUT         (),

    .EN_VTC           (1'b1)

    );

    end

end

else if(FAMILY == "7SERIES") begin : SERIES7_FAMILY

 

IDELAYCTRL  #(

.SIM_DEVICE ("7SERIES")

)

idelayctrl_inst(

.RDY       (idelayctrl_ready),

.REFCLK    (I_refclk),

.RST       (idelayctrl_reset)

);

 

    genvar k;

    for (k=0; k<DATA_WIDTH ; k=k+1)

    begin : rxdata_in_bus

    IDELAYE2 #(

    .HIGH_PERFORMANCE_MODE("TRUE"),

    .IDELAY_TYPE   ("FIXED"),

    .IDELAY_VALUE  (DELAY_SETA)

    )

    IDELAYE_A

    (

    .IDATAIN       (da_buf[k]),

    .DATAOUT       (da_buf_delay[k]),

    .DATAIN        (1'b0),

    .C             (1'b0),

    .CE            (1'b0),

    .INC           (1'b0),

    .CINVCTRL      (1'b0),

    .CNTVALUEIN    (5'h0),

    .CNTVALUEOUT   (),

    .LD            (1'b0),

    .LDPIPEEN      (1'b0),

    .REGRST        (1'b0)

    );

 

    IDELAYE2 #(

    .HIGH_PERFORMANCE_MODE("TRUE"),

    .IDELAY_TYPE   ("FIXED"),

    .IDELAY_VALUE  (DELAY_SETB)

    )

    IDELAYE_B

    (

    .IDATAIN       (db_buf[k]),

    .DATAOUT       (db_buf_delay[k]),

    .DATAIN        (1'b0),

    .C             (1'b0),

    .CE            (1'b0),

    .INC           (1'b0),

    .CINVCTRL      (1'b0),

    .CNTVALUEIN    (5'h0),

    .CNTVALUEOUT   (),

    .LD            (1'b0),

    .LDPIPEEN      (1'b0),

    .REGRST        (1'b0)

    );

    end

end

endgenerate

 

assign O_ad_da = da_buf_delay;

assign O_ad_db = db_buf_delay;

 

endmodule

 

5.2顶层模块调用程序

以下代码中,需要注意,通过 DELAY_SETA 和 DELAY_SETB 设置最佳的数据延迟。

对于有符号数据,通过设置 I_wave1_data(ads422x_da[11:4]+8'h80),加上 8'h80 让波形数据转为无符号,

在显示器上显示。

对于 ADS4225 修改 PLL 输出 125M 时钟

对于 ADS4229 修改 PLL 输出 250M 时钟

 

/**********************AD9248 ADC采集波形显示*************************

*********************************************************************/

 

`timescale 1ns / 1ns//仿真时间刻度/精度

 

module ads422x_top

(

input           I_sysclk_p,         //系统时钟输入

output          O_ads422x_clk_p,  

output          O_ads422x_clk_n,  

input           I_ads422x_clk,

input  [11 :0]  I_ads422x_da,

input  [11 :0]  I_ads422x_db,

output          O_ads422x_reset,

output          O_ads422x_sen,

output          O_ads422x_sclk,

output          O_card_power_en,    //子卡电源使能

 

output          O_HDMI_CLK_P,       //HDMI时钟输出 P端

output          O_HDMI_CLK_N,       //HDMI时钟输出 N端

output [2:0]    O_HDMI_TX_P,        //HDMI数据输出 P端

output [2:0]    O_HDMI_TX_N         //HDMI数据输出 N端

);

 

wire pclkx1,pclkx5,ref_clk,adc_clk,locked; //MMCM/PLL时钟信号  

 

//ADS422X直流版需要提供差分时钟工作

OBUFDS ADC_DIFF_CLK_O_inst (

.O (O_ads422x_clk_p),     // Diff_p output (connect directly to top-level port)

.OB(O_ads422x_clk_n),     // Diff_n output (connect directly to top-level port)

.I (adc_clk)         // Buffer input

);

 

//例化MMCM/PLL IP

clk_wiz_1 clk_hdmi_pll_inst

(

.clk_in1 (I_sysclk_p),

.reset   (!rst_cnt[7]),

.locked  (locked),

.clk_out1(ref_clk),

.clk_out2(pclkx1),//像素时钟

.clk_out3(pclkx5),//HDMI输出5倍像素时钟

.clk_out4(adc_clk)//输出给ADC

);

 

wire  [11 :0]  ads422x_da;

wire  [11 :0]  ads422x_db;

 

uiads422X_parallel #

(

.FAMILY    ("7SERIES"),

.DATA_WIDTH(12       ),//ADC数据位宽

.DELAY_SETA(25       ),//通道A的delay延迟

.DELAY_SETB(25       ) //通道B的delay延迟

)

uiads422X_parallel_inst

(

.I_refclk  (ref_clk), //IP内部的delay模块参考时钟,7系列200M KU KU+ 300M    

.I_reset   (~locked), //IP内部复位模块

.I_ad_clk  (I_ads422x_clk),//ADC同步时钟

.I_ad_da   (I_ads422x_da), //ADC数据输入通道A

.I_ad_db   (I_ads422x_db), //ADC数据输入通道B

.O_ad_da   (ads422x_da  ), //ADC数据输入数据经过延迟模块后输出,通道A

.O_ad_db   (ads422x_db  ), //ADC数据输入数据经过延迟模块后输出,通道B

.O_ad_reset(O_ads422x_reset), //ADC控制信号

.O_ad_sen  (O_ads422x_sen),    //ADC控制信号,设置ADC的工作模式

.O_ad_sclk (O_ads422x_sclk),  //ADC控制信号,设置ADC的工作模式

.O_ad_card_en(O_card_power_en)//ADC模块的电源使能,当用到电源使能的时候需要用到  

);

 

wire vtc_rstn,vtc_clk,vtc_vs,vtc_hs,vtc_de,vtc2_de;

wire [23:0] wave_rgb; //RGB颜色寄存器

assign vtc_clk  = pclkx1;//像素时钟

assign vtc_rstn = locked;//VTC复位信号

       

//上电延迟复位

reg [7:0]    rst_cnt=0; //复位计数器

wire  rstn = rst_cnt[7];//用高位复位

 

always @(posedge I_sysclk_p)begin

    if (rst_cnt[7])

        rst_cnt <=  rst_cnt;

    else

        rst_cnt <= rst_cnt+1'b1;

end

 

//例化HDMI 输出IP

uihdmitx #

(

.FAMILY("7FAMILY")  //选择芯片所支持的系列"7FAMILY" "UFAMILY"            

)

uihdmitx_inst

(

.I_rstn         (locked),//复位

.I_HS           (vtc_hs),//hs信号

.I_VS           (vtc_vs),//vs信号

.I_VDE          (vtc_de),//de信号

.I_RGB          (wave_rgb),//RGB数据

.I_PCLKX1       (pclkx1),//像素时钟

.I_PCLKX2_5     (1'b0),//2.5倍像素时钟,只有UFAMILY需要

.I_PCLKX5       (pclkx5),//5倍像素时钟

.O_TMDS_TX_CLK_P(O_HDMI_CLK_P),//HDMI时钟输出P端

.O_TMDS_TX_CLK_N(O_HDMI_CLK_N),//HDMI时钟输出N端

.O_TMDS_TX_P    (O_HDMI_TX_P),//HDMI输出数据P端

.O_TMDS_TX_N    (O_HDMI_TX_N)//HDMI输出数据N端

);

 

//此VTC IP 用于产生绘制波形的有效区域,波形绘制区域大小未1024*600

uivtc#

(

.H_ActiveSize(1280),          //视频时间参数,行视频信号,一行有效(需要显示的部分)像素所占的时钟数,一个时钟对应一个有效像素

.H_SyncStart(1280+88),        //视频时间参数,行同步开始,即多少时钟数后开始产生行同步信号

.H_SyncEnd(1280+88+44),       //视频时间参数,行同步结束,即多少时钟数后停止产生行同步信号,之后就是行有效数据部分

.H_FrameSize(1280+88+44+239), //视频时间参数,行视频信号,一行视频信号总计占用的时钟数

.V_ActiveSize(720),           //视频时间参数,场视频信号,一帧图像所占用的有效(需要显示的部分)行数量,通常说的视频分辨率即H_ActiveSize*V_ActiveSize

.V_SyncStart(720+4),          //视频时间参数,场同步开始,即多少行数后开始产生场同步信号

.V_SyncEnd (720+4+5),         //视频时间参数,场同步结束,即多少场数后停止产生场同步信号,之后就是场有效数据部分

.V_FrameSize(720+4+5+28),     //视频时间参数,场视频信号,一帧视频信号总计占用的行数量

.H2_ActiveSize(1024),         //波形绘制区域行像素大小        

.V2_ActiveSize(256)           //波形绘制区域场像素大小

)

uivtc_inst

(

.I_vtc_clk   (vtc_clk),  //系统时钟

.I_vtc_rstn  (vtc_rstn), //系统复位

.I_vtc2_offset_x(128),   //X坐标相对屏幕的原始坐标的偏移

.I_vtc2_offset_y(200),   //Y坐标相对屏幕的原始坐标的偏移

.O_vtc_vs    (vtc_vs),   //场同步输出

.O_vtc_hs    (vtc_hs),   //行同步输出

.O_vtc_de    (vtc_de),   //视频数据有效

.O_vtc2_de   (vtc2_de)   //绘制波形显示区域的有效区域

);

 

ila_0 ila_dbg (

.clk(I_ads422x_clk), // input wire clk

.probe0({ads422x_da,ads422x_db})

);

 

//测试数据产生,通过test_data产生测试数据,可以用于测试波形显示器的基本功能测试

//reg [7:0]test_data =0;

//always @(posedge vtc_clk)

//  if(vtc2_de)

//       test_data[7:0] = test_data + 1'b1;

 

//例化波形显示器 IP,默认支持2个通道数据,可以扩展支持更多通道

uiwave uiwave_inst

(

//波形1

.I_wave1_clk(I_ads422x_clk),//系统时钟输入

.I_wave1_data(ads422x_da[11:4]+8'h80),//ADC只显示高8bits 数据

.I_wave1_data_de(1'b1),//ADC数据有效信号

 

//波形2

.I_wave2_clk(I_ads422x_clk),//系统时钟输入

.I_wave2_data(ads422x_db[11:4]+8'h80),//ADC只显示高8bits 数据

.I_wave2_data_de(1'b1),//ADC数据有效信号

 

.I_vtc_rstn(vtc_rstn),//时序发生复位

.I_vtc_clk (vtc_clk), //像素时钟

.I_vtc_vs  (vtc_vs),  //场同步输出

.I_vtc_de  (vtc2_de),//同步,绘制波形显示区域的有效区域

.O_vtc_rgb (wave_rgb)//同步RGB数据 绘制数据输出  

     

);

 

endmodule

 

 

6测试结果

6.1硬件接线

(该教程使用的是1.8V的422X子卡,测试前请确保您手上的核心板电压已调整为1.8V。)

请确保下载器和开发板已经正确连接,并且开发板已经上电(注意JTAG端子不支持热插拔,而USB接口支持,所以在不通电的情况下接通好JTAG后,再插入USB到电脑,之后再上电,以免造成JTAG IO损坏)

 

 

 

6.2测试结果

 

逻辑分析仪采集的结果

标签:11,reset,vtc,clk,ads422x,cnt,b0,ADC,FEP
From: https://www.cnblogs.com/milianke/p/17931283.html

相关文章

  • 09 ADC模块FEP-DAQ7606采集显示波形方案
    软件版本:VIVADO2021.1操作系统:WIN1064bit硬件平台:适用XILINXA7/K7/Z7/ZU/KU系列FPGA登录米联客(MiLianKe)FPGA社区-www.uisrc.com观看免费视频课程、在线答疑解惑!1概述本方案通过把DAQ7606采集到的数据,通过前面已经完成的示波器显示驱动进行在屏幕上显示ADC采集的波形数......
  • 10 ADC模块FEP-DAQ9248采集显示波形方案
    软件版本:VIVADO2021.1操作系统:WIN1064bit硬件平台:适用XILINXA7/K7/Z7/ZU/KU系列FPGA登录米联客(MiLianKe)FPGA社区-www.uisrc.com观看免费视频课程、在线答疑解惑!1概述本方案通过把DAQ9248采集到的数据,通过示波器显示驱动进行在屏幕上显示ADC采集的波形数据。2系统框图......
  • [LeetCode Hot 100] LeetCode110. 平衡二叉树
    题目描述思路LeetCode104.二叉树的最大深度变种方法一:后序遍历(递归、dfs)/***Definitionforabinarytreenode.*publicclassTreeNode{*intval;*TreeNodeleft;*TreeNoderight;*TreeNode(){}*TreeNode(intval){this.......
  • [LeetCode Hot 100] LeetCode111. 二叉树的最小深度
    题目描述思路二叉树的最小深度就是第一个叶子节点所在的层数方法一:前序遍历(递归、dfs)/***Definitionforabinarytreenode.*publicclassTreeNode{*intval;*TreeNodeleft;*TreeNoderight;*TreeNode(){}*TreeNode(intva......
  • 还原win11右键菜单
    通过注册表修改win+R输入:regedit,打开注册表编辑器定位到:HKEY_CURRENT_USER\SOFTWARE\CLASSES\CLSID接着,右键点击CLSID键值,新建一个名为{86ca1aa0-34aa-4e8b-a509-50c905bae2a2}的项右键点击新创建的项,新建一个名为InprocServer32的项,按下回车键保存最后选择新创建的项,然后双......
  • 11-MySQL 存储引擎
    MySQL存储引擎可以理解为,MySQL的“文件系统”,(插件形式存在)只不过功能更加强大。mysql提供的存储引擎有InnoDB、MyISAM等TokuDB是第三方的存储引擎moardb,数据压缩比打,写入数据快,如果你的应用是读多写少的情况强烈建议使用此存储引擎innodb存储引擎的功能一、查看存储引擎1......
  • Oracle 11g RAC(openfiler +multipath +udev +补丁升级)
    配套视频:《Oracle11gRAC安装》 或《Oracle11gRAC安装》配套文档:《一步一步在linux上部署Oracle11gR2RAC》......
  • Windows11 win11提示这台电脑不符合安装此版本的Windows所需的最低系统要求怎么解决?
    Windows11win11提示这台电脑不符合安装此版本的Windows所需的最低系统要求怎么解决?  现在很多用户都会选择用U盘来安装系统,最新有用户在使用U盘安装Win11系统的时候,结果安装到第一步就提示这台电脑无法运行Windows11,这台电脑不符合安装此版本的Windows所需的最低系统要求。......
  • rust call sqlite3 error: linking with `link.exe` failed: exit code: 1181
    rustcallsqlite3error:linkingwithlink.exefailed:exitcode:1181声明:本文禁止csdn.net及所有所有子网站转载。禁止以营利性为目的的转载。报错error:linkingwith`link.exe`failed:exitcode:1181......
  • android-studio-2021.1.1.11-windows 版本遇到的各种大坑
    1、使用2023.12月的最新版本会无法选择java语言开发,所以必须选择较老的版本,比如我选择的android-studiobumbblebee 2021.1.1.11这个版本就可以选择java语言。2、不光需要设置代码自动补全提示,更重要的是最新的SDK,androidapi34有bug,无法语法提示,也可能是不支持我所使用的语......