

## 1 ACM2108/108 数据采集 DDR3 缓存网口发送实验

该文档主要是以 ACM2108 为例进行说明的，用户如果使用的 ACM108 模块，实验操作步骤同样适用，但是需要注意的是 ACM108 模块只有一个通道 0 可以使用。可将模块顶层的通道 1 注释，并更改 ACM108 对应引脚即可。系

本次实验，我们将通过电脑上的网络调试助手，将命令帧进行发送，然后通过 ACG525 开发板上的以太网芯片接收，随后将接收到的数据转换成命令，从而实现对 ACM2108/108 模块采样频率、数据采样个数以及采样通道的配置。配置完成之后，ACM2108/108 模块开始采集数据，将采集的数据存储至 DDR 中最后数据通过网口传输至电脑，电脑端将采集到的数据通过 MATLAB 进行进一步的分析，系统的整体设计框图如图 1-1 所示。



图 1-1 ACM2108/108 数据采集 DDR3 缓存网口发送整体设计框图

模块说明如下：

1. pll 模块：锁相环模块，输入时钟 50M；输出 50M 的时钟给其它模块使用；输出 125M 的时钟给 DDS 模块使用；输出 50M 时钟，时钟相位偏移负 90 度的 AD\_Clk 给 ACM2108/ACM108 模块使用。
  2. eth\_cmd 模块：以太网接收命令模块，接收以太网的数据并输出到对应的模块。
  3. cmd\_rx 模块：将以太网的数据解析为对应指令并输出到对应的模块。
  4. speed\_ctrl 模块：根据解析的指令，控制采样的速度。

5. ad\_8bit\_to\_16bit 模块：将 acm2108 采样的 8 位数据转换成 16 位的数据，方便给上位机进行分析。
6. acm2108\_test 模块：ACM2108 的 DAC 控制器模块，根据按键控制 DAC 生成的波形和频率。
7. state\_ctrl 模块：ADC 采集数据 DDR3 缓存以太网发送状态控制模块，协调各个模块的信号控制，程序状态的总控制模块。
8. ddr3\_ctrl\_2port 模块：fifo 接口到 ddr3 接口的转换模块(含 2 个 FIFO)。

## 1.1 ACM2108 高速 ADC+DAC 模块简介

本模块基于国产知名模拟器件设计和制造商杭州瑞盟公司的 8 位 50M 采样速率高速 ADC 芯片 MS9280（完全兼容 AD9280，但是速度比 AD9280 高）和 125Msps 转换速率的高速 DAC 芯片 MS9708（完全兼容 AD9708）。配合前端模拟信号调理电路，实现了±5V 电压范围内信号的高速采样，以及模拟信号输出。下图 1-2 为模块实物图。



图 1-2 ACM2108 模块实物图

本模块可用于小梅哥全系列 FPGA、SOC、Zynq 开发板，包括国产开发板和各核心板的评估底板。AC620、AC6102、ACX720、ACZ702、AC609、智多晶 FPGA 开发板（AC208-SA5Z）、AC608 评估底板、AC601 评估底板、AC675 评估底板等。

## 1.2 模块设计

### 1.2.1 eth\_cmd 模块

接收转命令模块 eth\_cmd，将以太网传输过来的指令数据帧进行拆解，得到需要的指令数据传送给别的模块进行处理，该模块的结构框图如下图 1-3 所示。



图 1-3 接收转命令模块框图

该模块的信号说明如下表 1-1 所示。

表 1-1 接收转命令模块信号说明表

| 信号名称           | I/O | 信号意义              |
|----------------|-----|-------------------|
| clk            | I   | 模块工作时钟            |
| reset_n        | I   | 模块复位信号，低电平有效      |
| fifodout[7:0]  | I   | 从 FIFO 中读出的 8 位数据 |
| rx_empty       | I   | FIFO 为空标志信号       |
| fifo_rd_req    | O   | FIFO 的读请求信号       |
| cmdvalid       | O   | 输出命令有效标志信号        |
| address[7:0]   | O   | 配置 ADC 的寄存器地址信号   |
| cmd_data[31:0] | O   | 写入到寄存器中的数据        |

网口一次发送的命令数据内容为 32 个字节，为了实现通过网口修改这些寄存器的值，需要对发送一次的数据进行拆解才能实现，对于设计的数据帧，一帧数据一共 8 个字节，包含帧头、帧尾、地址段、数据段。帧格式如下表 1-2 所示：

表 1-2 帧格式说明表

| 数据 | D0   | D1   | D2         | D3          | D4           | D5         | D6        | D7   |
|----|------|------|------------|-------------|--------------|------------|-----------|------|
| 功能 | 帧头 0 | 帧头 1 | 地址 address | data[31:24] | data[23: 16] | data[15:8] | data[7:0] | 帧尾   |
| 值  | 0x55 | 0xA5 | XX         | XX          | XX           | XX         | XX        | 0xF0 |

从上表中可以看出，每帧数据一共 8 个字节，分别用 D0~D7 表示，其中，D0 和 D1 两个数据作为帧头，其值固定为 0x55、0xA5，D7 作为帧尾，其值固

定为 0xF0。帧头和帧尾的作用是为了准确识别数据帧，确保接收的数据是我们需要分析的。D2 代表的是要操作的寄存器地址，D3 为要写入寄存器的数据的 24~31 位，D4 为要写入寄存器的数据的 16~24 位，D5 为要写入寄存器的数据的 8~15 位，D6 为要写入寄存器的数据的 0~7 位。

该模块的作用就是将网口接收到的数据拆解成上述帧格式，将 D2 作为地址 address 输出，指定修改哪个寄存器，D3~D6 共 32 位作为数据 data 输出，控制 ACM2108/108 模块进行相应的配置。下面将对模块中的部分代码进行说明：

首先，产生 FIFO 的读请求信号。当检测到 FIFO 非空的时候，产生 FIFO 读请求信号，代码如下所示：

```
always@(posedge clk or negedge reset_n)
if(!reset_n)
    fifo_rd_req <= 1'b0;
else if(!rx_empty)
    fifo_rd_req <= 1'b1;
else
    fifo_rd_req <= 1'b0;
```

然后是得到帧命令数据，每产生一次读请求，将会从 FIFO 中读取一个 8 位的数据，连续存储 8 个字节的数据就得到一帧命令数据。代码如下所示：

```
always@(posedge clk)
if(fifo_rd_req)begin
    data_0[7] <= #1 fifodout;
    data_0[6] <= #1 data_0[7];
    data_0[5] <= #1 data_0[6];
    data_0[4] <= #1 data_0[5];
    data_0[3] <= #1 data_0[4];
    data_0[2] <= #1 data_0[3];
    data_0[1] <= #1 data_0[2];
    data_0[0] <= #1 data_0[1];
end
```

最后是判断得到的帧命令数据是否正确，当数据符合 D0 为 8'h55，D1 为 8'hA5，D7 为 8'hF0，则代表该数据格式正确，会生成一个指令正确信号 cmdvalid 输出到指令转控制模块，并将数据进行输出，代码如下所示：

```
always@(posedge clk or negedge reset_n)
if(!reset_n)begin
    address <= 0;
    cmd_data <= 32'd0;
    cmdvalid <= 1'b0;
end
else if(fifo_rx_done)begin
    if((data_0[0]==8'h55)&&(data_0[1]==8'hA5)&&(data_0[7]==8'hF0))
```

```

begin
    cmd_data[7:0] <= #1 data_0[6];
    cmd_data[15:8] <= #1 data_0[5];
    cmd_data[23:16] <= #1 data_0[4];
    cmd_data[31:24] <= #1 data_0[3];
    address <= #1 data_0[2];
    cmdvalid <= #1 1;
end
else
    cmdvalid <= #1 0;
end
else
    cmdvalid <= #1 0;

```

## 1.2.2 cmd\_rx 模块

指令转控制模块 cmd\_rx 将从接收转命令模块接收到的数据转换为相应的控制数据，首先将对寄存器进行说明，其功能和地址分别如表 1-3 所示：

表 1-3 寄存器说明表

| 名称            | 地址 | 位宽 | 功能简介                                                                                                                                                                             |
|---------------|----|----|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| RestartReq    | 0  | 1  | 重新开始采集请求寄存器，向该寄存器写入任意值即可启动新一轮的采样存储传输                                                                                                                                             |
| ChannelSel    | 1  | 2  | 通道设置寄存器，共 2 位。ACM2108 模块提供了 ADC0、ADC1 两个通道进行数据采集，ACM108 模块则只有一个 ADC0 通道可以用。                                                                                                      |
| DataNum       | 2  | 32 | 数据个数寄存器。如果采样 512 个数据，应该向寄存器中写入 01 00。                                                                                                                                            |
| ADC_Speed_Set | 3  | 32 | ADC 采样速率设置寄存器。如果设置为 0，采样和时钟保持一致 50M 时钟就是 50M 的采样速率，设置计数值后就可以改变采样频率，设置为 1 就是 25M。如果设置为 27 0F，换算成十进制是 9999，采样速率设置是 5k，计数值和采样频率之间的关系：设置计数值 = Fclk/Fs - 1，Fs 是期望的采样率，Fclk 是系统时钟 50M。 |

指令转控制模块的结构框图如图 1-4 所示。



图 1-4 指令转控制模块结构框图

模块信号说明如表 1-4 所示。

表 1-4 指令转控制模块信号说明表

| 信号名称                | I/O | 信号意义          |
|---------------------|-----|---------------|
| clk                 | I   | 模块时钟信号        |
| reset_n             | I   | 模块复位信号，低电平有效  |
| cmd_data[31:0]      | I   | 写入到寄存器中的值     |
| cmdvalid            | I   | 命令有效标志信号      |
| cmd_addr[7:0]       | I   | 寄存器地址信号       |
| ChannelSel[1:0]     | O   | 通道设置寄存器       |
| DataNum[31:0]       | O   | 数据个数寄存器       |
| ADC_Speed_Set[31:0] | O   | ADC 采样速率控制寄存器 |
| RestartReq          | O   | 重新开始采集请求信号    |

根据表 1-4 中的内容，地址 cmd\_addr 为 0 时，产生 RestartReq； cmd\_addr 为 1 时，得到通道设置数据 cmd\_data[1:0]； cmd\_addr 为 2 时，得到需要采样的数量 cmd\_data[31:0]； cmd\_addr 为 3 时，得到设置的采样速率的值 cmd\_data[31:0]，代码如下所示：

```
always@(posedge clk or negedge reset_n)
begin
    if(!reset_n)begin
        ChannelSel <= 2'b00;
        DataNum <= 32'd0;
        ADC_Speed_Set <= 32'd0;
        RestartReq <= 1'b0;
    end
    else if(cmdvalid)begin
        case(cmd_addr)
            0: RestartReq <= 1'b1;
            1: ChannelSel <= cmd_data[1:0];
            2: DataNum <= cmd_data[31:0];
            3: ADC_Speed_Set <= cmd_data[31:0];
        default:;
        endcase
    end
    else
        RestartReq <= 1'b0;
end
```

### 1.2.3 acm2108\_test 模块

ACM2108 DAC 波形控制模块（acm2108\_test）的功能是 DAC 输出信号波形，并通过按键切换信号类型。

该模块的基本结构框图如下图 1-5 所示：



图 1-5 acm2108\_ctrl 模块基本结构框图

对上述图中模块的信号说明如下表 1-5 所示：

表 1-5 acm2108\_ctrl 模块信号说明表

| 信号名称          | I/O | 信号意义                   |
|---------------|-----|------------------------|
| Clk           | I   | 模块时钟信号, 50M 时钟         |
| reset_n       | I   | 模块复位信号, 低电平有效          |
| key_in        | I   | 按键输入信号                 |
| AD0[7:0]      | I   | ADC 模块 ADC0 输入的 8 位数据  |
| AD1[7:0]      | I   | ADC 模块 ADC1 输入的 8 位数据  |
| clk125m       | I   | 输入的 125M 的时钟           |
| AD_Clk        | I   | 输入的 ADC 时钟             |
| AD0_CLK       | O   | ADC 模块的 ADC0 工作时钟信号    |
| AD1_CLK       | O   | ADC 模块的 ADC1 工作时钟信号    |
| DA0_Data[7:0] | O   | ADC 模块的 DAC0 的 8 位数据输出 |
| DA1_Data[7:0] | O   | ADC 模块的 DAC1 的 8 位数据输出 |
| DA0_Clk       | O   | ADC 模块的 DAC0 工作时钟信号    |
| DA1_Clk       | O   | ADC 模块的 DAC1 工作时钟信号    |

## 1.2.4 speed\_ctrl 模块

采样速率控制 (speed\_ctrl) 模块用来控制 ACM2108/108 的采样速率，该模块的结构框图如下图 1-6 所示。



图 1-6 采样速率控制模块

对该模块的信号说明如下表 1-6 所示。

表 1-6 采样速率控制模块信号说明表

| 信号名称          | I/O | 信号意义                                                                                      |
|---------------|-----|-------------------------------------------------------------------------------------------|
| clk           | I   | 模块时钟信号                                                                                    |
| reset_n       | I   | 模块复位信号, 低电平有效                                                                             |
| ad_sample_en  | I   | 输入的启动采样标志信号                                                                               |
| div_set[31:0] | I   | 采样频率数据控制信号, $\text{div\_set} = \text{Fclk}/\text{Fs} - 1$ ,<br>Fs 是期望的采样率, Fclk 是系统时钟 50M |
| adc_data_en   | O   | ADC 采样结果存储使能信号                                                                            |

ACM2108/108 模块的最大采样速率为 50M, 如需使用低于时钟频率的采样速率, 可以依旧给 ADC 提供 50MHz 的时钟信号, 但在 FPGA 内部, 对 50Msps 的采样结果数据进行抽取重采样的方法实现。比如期望以 1Msps 的采样速率采样, 则只需要每间隔 50 个采样数据取一个结果存储或使用, 其他 49 个数据直接舍弃, 这样就能实现 1MSPS 的采样率了。下面我们将编写相应代码实现上述功能。

设置一个计数器 div\_cnt, 当产生采样使能信号 ad\_sample\_en 之后, 计数器加 1, 当计数值等于设置的 div\_set 的时候, 将计数器清零。代码如下所示:

```

always@(posedge clk or negedge reset_n)
if(!reset_n)
    div_cnt <= 0;
else if(ad_sample_en)begin
    if(div_cnt >= div_set)
        div_cnt <= 0;
    else
        div_cnt <= div_cnt + 1'd1;
end
else

```

```
div_cnt <= 0;
```

计数器的计数值达到 div\_set 的时候，使能 ADC 采样结果存储使能信号 adc\_data\_en，我们将该信号输出，最终实现每隔 div\_set 个采样数据取一个结果存储或使用，从而达到对 ADC 采样频率的控制。代码如下所示：

```
always@(posedge clk or negedge reset_n)
if(!reset_n)
    adc_data_en <= 0;
else if(div_cnt == div_set)
    adc_data_en <= 1;
else
    adc_data_en <= 0;
```

## 1.2.5 8bit 转 16bit 模块

为了方便后续进行上位机进行数据采集，将 8bit 数据转换成 16bit 数据进行存储。8bit 转 16bit 模块的基本框图如下图 1-7 所示，端口列表如下表 1-7 所示。



图 1-7 8bit 转 16bit 模块的基本框图

表 1-7 8bit 转 16bit 模块的端口列表

| 信号名称         | I/O | 信号意义                 |
|--------------|-----|----------------------|
| clk          | I   | 模块时钟信号               |
| ad_sample_en | I   | 数据采集使能信号             |
| ch_sel[1:0]  | I   | 采样通道命令信号             |
| AD0[7:0]     | I   | ADC 通道 0 的 8 位数据输入信号 |
| AD1[7:0]     | I   | ADC 通道 1 的 8 位数据输入信号 |
| ad_out       | O   | 16 位数据输出信号           |
| ad_out_valid | O   | 输出数据有效信号             |

这里我们给出代码如下：

```
module ad_8bit_to_16bit(
    clk,
    ad_sample_en,
    ch_sel,
    AD0,
```

```
AD1,
ad_out,
ad_out_valid
);









else if(ad_sample_en && ch_sel == 2'b10)
ad_out<={4'd0,s_ad_in2,4'd0};//  

else if(ad_sample_en && ch_sel == 2'b00)
ad_out<={4'd0,adc_test_data,4'd0};  

else
ad_out <= 16'd0;

always @(posedge clk)
ad_out_valid <= ad_sample_en;

endmodule
```

ACM2108/108 模块采集到的数据是无符号的数据，比如采集的波形为 +5V~−5V 的正弦波，ADC 模块最终输出的数据就为 255~0，而我们上位机在分析数据的时候需要数据是有符号的，在这里我们进行的操作就是将 ADC 采集得到的数据加上 128，也就是将最高位取反，最后进行分析时将最高位作为符号位。

## 1.2.6 state\_ctrl 模块

DDR3 双端口转换模块控制信号的产生以及 ADC 何时启动数据传输都是通过 state\_ctrl 模块控制的，该模块的结构框图如下图 1-8 所示。



图 1-8 state\_ctrl 模块结构框图

该模块的信号说明如下表 1-8 所示。

表 1-8 state\_ctrl 模块信号说明表

| 信号名称                  | I/O | 信号意义                                                                            |
|-----------------------|-----|---------------------------------------------------------------------------------|
| clk                   | I   | 模块工作时钟                                                                          |
| reset                 | I   | 模块复位信号，高电平有效                                                                    |
| start_sample          | I   | ADC 模块开始采样标志信号                                                                  |
| set_sample_num[31:0]  | I   | 设置的采样个数                                                                         |
| rdfifo_empty          | I   | 读 FIFO 的读空标识信号，用于标识当前 FIFO 是否为空（即 FIFO 内有无数据）                                   |
| rdfifo_dout[15:0]     | I   | 读 FIFO 的数据输出，数据位宽为 16 位                                                         |
| wrfifo_full           | I   | 写 FIFO 的写满标识信号，用于标识当前 FIFO 是否有被写满                                               |
| adc_data_en           | I   | ADC 采样结果存储使能信号                                                                  |
| wrfifo_clr            | O   | FIFO 的写清除信号，wrfifo_clr 向外打三拍输出，保证 wrfifo 的清零信号的生效节拍数                            |
| rdfifo_clr            | O   | 读 FIFO 清空控制信号，给高电平表示执行清空，执行清空操作时，需保证给 3 个及以上个时钟（rdfifo_clk）周期的高电平               |
| rdfifo_rden           | O   | 读 FIFO 的读数据使能控制信号，给高电平表示往 FIFO 读数据，为避免读数据的丢失，确保在 FIFO 非空（rdfifo_empty =0）情况下读数据 |
| ad_sample_en          | O   | ADC 采样使能标志信号                                                                    |
| eth_fifo_wrreq        | O   | 以太网发送缓存 FIFO 的写请求信号                                                             |
| eth_fifo_wrdata[15:0] | O   | 写入以太网发送缓存 FIFO 的 16 位数据                                                         |

fifo\_axi4\_adapter 模块需要的控制信号有：wrfifo\_clr、wrfifo\_clk、wrfifo\_wren、wrfifo\_din、rdfifo\_clr、rdfifo\_clk、rdfifo\_rden。上述信号中：wrfifo\_clk、rdfifo\_clk 应该与该模块的工作时钟保持一致，也就是和 ADC 数据采集模块的工作时钟保持一致为 50M；wrfifo\_wren 信号为 ADC 数据位扩展模块输出数据有效信号 ad\_out\_valid；wrfifo\_din 信号是 ADC 控制模块输出数据信号 ad\_out，除去上述已经得到的控制信号外，state\_ctrl 模块还需要产生的控制信号包括：wrfifo\_clr、rdfifo\_clr、rdfifo\_rden。

根据上述描述，我们可以通过状态机的方式实现该模块的功能，状态定义如下所示：

|                                     |         |                   |
|-------------------------------------|---------|-------------------|
| <b>localparam</b> IDLE              | = 4'd0; | //空闲状态            |
| <b>localparam</b> DDR_WR_FIFO_CLEAR | = 4'd1; | //DDR 写 FIFO 清除状态 |
| <b>localparam</b> ADC_SAMPLE        | = 4'd2; | //ADC 采样数据状态      |
| <b>localparam</b> DDR_RD_FIFO_CLEAR | = 4'd3; | //DDR 读 FIFO 清除状态 |
| <b>localparam</b> DATA_SEND_START   | = 4'd4; | //数据发送状态          |
| <b>localparam</b> DATA_SEND_WORKING | = 4'd5; | //数据发送完成状态        |

下面对每个状态的实现进行说明。

### 1. IDLE 状态

当处于空闲状态时，对 start\_sample 采样起始位进行寄存，同时限定其只工作在状态 IDLE，代码如下所示：

```
always@(posedge clk or posedge reset)begin
if(reset)
    start_sample_rm <= 1'b0;
else if(state==IDLE)
    start_sample_rm <= start_sample;
else
    start_sample_rm <= 1'b0;
end
```

当产生 start\_sample\_rm 信号之后跳转到 DDR\_WR\_FIFO\_CLEAR 状态，代码如下所示：

```
begin
if(start_sample_rm) begin //DDR 初始化完成并且产生启动采样信号
    state <= DDR_WR_FIFO_CLEAR; //进入写 FIFO 清除状态
end
else begin
    state <= state;
end
end
```

### 2. DDR\_WR\_FIFO\_CLEAR 状态

当进入写 FIFO 清零状态后，开始清除写 FIFO 内的原始数据。设置清除 DDR 写 FIFO 的计数器，保证至少 10 拍的延时，代码如下所示：

```
always@(posedge clk or posedge reset)begin
if(reset)
    wrfifo_clr_cnt<=0;
else if(state==DDR_WR_FIFO_CLEAR)//如果进入了清 fifo 状态
begin
    if(wrfifo_clr_cnt==9)
        wrfifo_clr_cnt<=5'd9;
    else
        wrfifo_clr_cnt<=wrfifo_clr_cnt+1'b1;
end
else
    wrfifo_clr_cnt<=5'd0;
end
```

然后等待 wrfifo\_full（写 fifo 满信号）的信号拉低，拉低后，表示可以往 FIFO 里写入数据，此时进入下一个状态。在清空（复位）FIFO 的时候，FIFO 的 full 信号会变高，可以认为在复位 FIFO 时是不允许对 FIFO 进行写操作的，即使写也是不可靠的，等 FIFO 的复位结束后，full 信号会变低，就允许对 FIFO 进行写操作。写 FIFO 清除状态代码如下所示：

```
begin
if(!wrfifo_full && (wrfifo_clr_cnt==9))
    state<=ADC_SAMPLE;
else
    state<=DDR_WR_FIFO_CLEAR;
end
```

当处于 DDR\_WR\_FIFO\_CLEAR 状态时，我们需要产生清除写 FIFO 的信号 wrfifo\_clr，由三拍延时信号拉高提供，之所以提供的延迟信号时间为 3 拍，是为了给清 FIFO 信号足够的拉高时间，以保证清空指令的可靠，也就是在 wrfifo\_clr\_cnt 为 0、1 或 2 时，wrfifo\_clr 置 1，否则 wrfifo\_clr 为 0。

```
always@(posedge clk or posedge reset)begin
if (reset)
    wrfifo_clr<=0;
else if(state==DDR_WR_FIFO_CLEAR)
begin
    if(wrfifo_clr_cnt==0||wrfifo_clr_cnt==1||wrfifo_clr_cnt==2)
        wrfifo_clr<=1'b1;
    else
        wrfifo_clr<=1'b0;
end
else
    wrfifo_clr<=1'b0;
```

```
end
```

### 3. ADC\_SAMPLE 状态

进入 ADC 采样数据状态之后，首先设置 ADC 采样个数计数器 adc\_sample\_cnt，然后计数器根据 ADC 输出数据使能信号 adc\_data\_en 进行计数，代码如下所示：

```
always@(posedge clk or posedge reset)begin
if(reset)
    adc_sample_cnt<=1'b0;
else if(state==ADC_SAMPLE)begin
    if(adc_data_en)
        adc_sample_cnt<=adc_sample_cnt+1'b1;
    else
        adc_sample_cnt<=adc_sample_cnt;
end
else
    adc_sample_cnt<=1'b0;
end
```

当 adc\_sample\_cnt 达到设定的采样数据个数，并且 adc\_data\_en 有效的时候时，ADC 模块数据采集完成，跳转到读 FIFO 清除状态，代码如下所示：

```
begin
if((adc_sample_cnt>=set_sample_num-1'b1)&& adc_data_en)
    state<=DDR_RD_FIFO_CLEAR;
else
    state<=state;
end
```

当处于 ADC\_SAMPLE 状态时，我们还需要产生采样使能信号 ad\_sample\_en 给到其他模块使用，代码如下所示：

```
always@(posedge clk or posedge reset)begin
if(reset)
    ad_sample_en<=0;
else if(state==ADC_SAMPLE)
    ad_sample_en<=1;
else
    ad_sample_en<=0;
end
```

### 4. DDR\_RD\_FIFO\_CLEAR 状态

进入清 FIFO 状态之后，首先设置清除读 FIFO 的计数器 rdfifo\_clr\_cnt，保证至少 10 拍的延时。代码如下所示：

```
always@(posedge clk or posedge reset)begin
if(reset)
```

```

    rdfifo_clr_cnt<=0;
else if(state==DDR_RD_FIFO_CLEAR)//如果进入了清 fifo 状态
begin
    if(rdfifo_clr_cnt==9)
        rdfifo_clr_cnt<=5'd9;
    else
        rdfifo_clr_cnt<=rdfifo_clr_cnt+1'b1;
end
else
    rdfifo_clr_cnt<=5'd0;
end

```

然后等待 `rdfifo_empty` (读端 FIFO 的空信号) 信号拉低, 拉低后, 表示 FIFO 里已经有被写入数据, 此时进入下一个状态。在清空(复位) FIFO 的时候, FIFO 的 `empty` 信号会变高, 可以认为在复位 FIFO 时是不允许对 FIFO 进行读操作的, 即使读也是不可靠的, 等 FIFO 的复位结束后, FIFO 被写入数据后, `empty` 信号会变低, 就允许对 FIFO 进行读操作, 然后跳转到 `DATA_SEND_START` 状态, 读 FIFO 清除状态代码如下所示:

```

begin
    if(!rdfifo_empty && (rdfifo_clr_cnt==9))begin
        state<=DATA_SEND_START;
    end
    else
        state<=state;
end

```

当处于 `DDR_WR_FIFO_CLEAR` 状态时, 我们需要产生清除读 FIFO 的信号 `rdfifo_clr`, 由三拍延时信号拉高提供, 之所以提供的延迟信号时间为 3 拍, 是为了给清 FIFO 信号足够的拉高时间, 以保证清空指令的可靠, 也就是在 `rdfifo_clr_cnt` 为 0、1 或 2 时, `rdfifo_clr` 置 1, 否则 `rdfifo_clr` 为 0。

```

always@(posedge clk or posedge reset)begin
if (reset)
    rdfifo_clr<=0;
else if(state==DDR_RD_FIFO_CLEAR)
begin
    if(rdfifo_clr_cnt==0||rdfifo_clr_cnt==1||rdfifo_clr_cnt==2)
        rdfifo_clr<=1'b1;
    else
        rdfifo_clr<=1'b0;
end
else
    rdfifo_clr<=1'b0;
end

```

## 5. DATA\_SEND\_START

进入DATA\_SEND\_START 状态之后，state 直接跳转到数据发送状态，启动传输，代码如下所示：

```
begin
    state <= DATA_SEND_WORKING;
end
```

## 6. DATA\_SEND\_WORKING 状态

进入数据发送状态之后，产生 rdfifo\_rden 信号，将 DDR 中的数据读取出来，当读出的数据达到需要采集的数据信号时，跳转到 IDLE 状态，完成一次数据传输，代码如下所示：

```
begin
    if(send_data_cnt >= set_sample_num-1'b1) begin
        rdfifo_rden <= 1'b0;
        state <= IDLE;
    end
    else begin
        rdfifo_rden <= 1'b1;
        state <= DATA_SEND_WORKING;
    end
end
```

send\_data\_cnt 在 rdfifo\_rden 有效的情况下进行计数，从而统计从 DDR 中读取的数据个数，代码如下所示：

```
always@(posedge clk or posedge reset)begin
    if(reset)
        send_data_cnt<=32'd0;
    else if(state==IDLE)
        send_data_cnt<=32'd0;
    else if(rdfifo_rden)
        send_data_cnt<=send_data_cnt+1;
    else
        send_data_cnt<=send_data_cnt;
end
```

最后，当 rdfifo\_rden 信号为高电平的时候，将 DDR 读出的数据存放至以太网发送缓存 FIFO 中，代码如下所示：

```
always@(posedge clk or posedge reset)
if(reset) begin
    eth_fifo_wrreq <= 1'b0;
    eth_fifo_wrdata <= 'd0;
end
else if(rdfifo_rden) begin
    eth_fifo_wrreq <= 1'b1;
    eth_fifo_wrdata <= rdfifo_dout;
```

```

end
else begin
    eth_fifo_wrreq <= 1'b0;
    eth_fifo_wrdata <= 'd0;
end

```

## 1.2.7 ddr3\_ctrl\_2port 模块

fifo 接口到 ddr3 接口的控制器模块（ddr3\_ctrl\_2port）的设计，读者可自行查看教程。

需要注意的是，Gowin\_V1.9.9 之后 DDR IP 新增加了一个 pll\_stop 信号，用来控制 DDR 时钟的启动。关于改信号的具体配置，可参考如下链接。

【高云 FPGA】更新 DDR IP 之后初始化不起来的问题解决方案

<https://fpga.cn/forum.php?mod=viewthread&tid=29815>

(出处: 芯路恒电子技术论坛)

### 1.2.7.1 eth\_send\_ctrl 模块

网口发送控制模块（eth\_send\_ctrl）主要负责配置控制网口发送模块的使能控制信号 pkt\_tx\_en，并通过 pkt\_length 信号对以太网数据帧数据长度进行控制，该模块的结构框图如下图 1-9 所示。



图 1-9 网口发送控制模块

对该模块的信号说明如下表 1-9 所示：

表 1-9 网口发送控制模块信号说明表

| 信号名称    | I/O | 信号意义                  |
|---------|-----|-----------------------|
| clk125M | I   | 模块时钟信号信号，以太网工作时钟 125M |

|                       |   |                |
|-----------------------|---|----------------|
| reset_n               | I | 模块复位信号, 低电平复位  |
| fifo_rd_cnt [10:0]    | I | FIFO 读数据计数     |
| total_data_num [31:0] | I | 需要采集的数据个数      |
| restart_req           | I | 开始采样请求信号       |
| eth_tx_done           | I | 以太网一个包传输完成标志信号 |
| pkt_length [15:0]     | O | 以太网需要传输的数据长度   |
| pkt_tx_en             | O | 以太网传输使能信号      |

以太网帧最大长度 1518 字节 (数据段 1500 字节), 其中数据段 1500 字节还包括 20 字节 IP 报文头部和 8 字节 UDP 报文头部, 所以数据帧发送的 ACM2108/108 采集的数据最大长度为 1472 字节。

该模块可以通过编写状态机代码的方式实现功能, 下面将对每个状态的代码进行介绍。

状态 0, 得到 pkt\_length 信号的初始值。这里需要注意的是经过数据位扩展模块输出的数据为 16 位的, 每个数据占据 2 个字节, 所以发送 N 个采样数据, 则以太网需要发送 2\*N 个字节数据。当产生开始采样请求 restart\_req 之后, 系统开始采样, 同时, 将需要采集的数据个数 total\_data\_num 左移一位 (相当于乘以 2), 得到实际需要以太网传输的数据, 当数据大于 16'd1472 时, 设置 pkt\_length 为最大传输长度 1472; 当数据大于 0 时, pkt\_length 等于为 total\_data\_num 乘以 2。给 pkt\_length 赋初值之后, 跳转到状态 1, 代码如下所示:

```
if(restart_req)begin
    data_num <= total_data_num;
    if((total_data_num << 1) >= 16'd1472)begin
        pkt_length <= 16'd1472; //一个数据 2 个字节
        state <= 1;
    end
    else if((total_data_num << 1) > 0)begin
        pkt_length <= total_data_num << 1; //一个数据 2 个字节
        state <= 1;
    end
    else begin
        state <= 0;
    end
end
```

状态 1, 当 FIFO 计数信号 fifo\_rd\_cnt 的数值满足一帧数据帧发送采集数据长度时, 产生 pkt\_tx\_en 信号, 以太网发送模块开始读取 FIFO 中的数据。代码如下所示:

```
if(fifo_rd_cnt >= (pkt_length -2))begin
    pkt_tx_en <= 1'b1;
    state <= 2;
end
```

```

else begin
    state <= 1;
    pkt_tx_en <= 1'b0;
end

```

状态 2，当以太网一个包传输完成之后，产生 eth\_tx\_done 信号，此时剩下需要传输的数据 data\_num 应该减去 pkt\_length 的一半，这是因为 ADC 采集的数据是 16 位的，但是以太网每次只传送 8 位数据，所以以太网实际传输的数据应该 ADC 采集到的数据的一半。代码如下所示：

```

begin
    pkt_tx_en <= 1'b0;
    if(eth_tx_done)begin
        data_num <= data_num - pkt_length/2;
        state <= 3;
    end
end

```

状态 3，设置以太网的帧间隙时间间隔。本次实验使用的千兆以太网，其相邻的两帧之间的最小间隔时间为 96ns，在设置的时候只要大于 96ns 就行，本次实验设置 128 个时钟周期，也就是 1024ns。当设置的时间比较小的时候，虽然以太网传输速率会加快，但是会增加电脑端解析数据包的压力，当时间过大，又会影响以太网传输速率，所以在设置的时候需要设定一个比较合理的值。

```

if(cnt_dly_time >= cnt_dly_min)begin
    state <= 4;
    cnt_dly_time <= 28'd0;
end
else begin
    cnt_dly_time <= cnt_dly_time + 1'b1;
    state <= 3;
end

```

状态 4，进行连续的包传输。当剩下的需要以太网传输的数据 (data\_num \* 2) 大于包传输最大数据 1472 时，使 pkt\_length 为 1472，然后回到状态 1 继续传输；当剩下需要传输的数据不足包传输最大数据 1472 时，pkt\_length 为剩下的需要传输的数据 data\_num \* 2，然后回到状态 1 传输剩下的数据。代码如下所示。

```

begin
    if(data_num * 2 >= 16'd1472)begin
        pkt_length <= 16'd1472;
        state <= 1;
    end
    else if(data_num * 2 > 0)begin
        pkt_length <= data_num * 2;
        state <= 1;
    end
end

```

```
else begin  
    state <= 0;  
end  
end
```

模块设计完成之后，只需要在顶层文件中对各个模块之间的接口信号进行连接，完整的顶层文件代码请自行查看例程文件。

## 1.3 板级验证

本次实验的板级验证环节，主要验证：通过电脑上的网络调试助手，将命令帧进行发送，然后通过 ACG525 开发板上的以太网芯片接收，随后将接收到的数据转换命令，最终实现对 ACM2108/108 模块的采样频率、数据采样个数以及采样通道的配置。配置完成之后，ACM2108/108 模块开始采集数据，将 ACM2108/108 模块采集的数据通过网口传输到电脑。电脑端将接收到的数据进行保存，然后通过 MATLAB 进行进一步的分析。针对本次实验，我们也提供有专门的上位机软件，用户只需要在软件界面进行参数配置，便可以实时观察到实时的数据波形变化，使用起来非常方便。

### 1.3.1 系统所需硬件

1. ACG525 开发板一块
2. 网线一根
3. ACM2108 模块一个或者 ACM108 模块一个
4. Type-C 下载线一根
5. 下载器一个
6. 电源线一根
7. 信号发生器一台

### 1.3.2 硬件连接

本次设计系统硬件连接：

1. 使用 下载器连接开发板 Jtag 接口和电脑 USB 口
2. 将开发板电源拨码开关拨到对应侧
3. 将网线连接至网口上

4. ACM2108 模块连接至 40 pin 的排针上，靠右连接，1 脚和 1 脚对应。

ACM2108 模块硬件连接如图 1-10 和所示



图 1-10 ACM2108 硬件连接图

### 1.3.3 修改电脑 IP 地址

本次实验设定了目标 IP 地址（PC 端）为 192.168.0.3，用户需要将自己电脑上的以太网 IP 地址修改为该地址，在本地连接状态中，点击属性，并在弹出的属性对话框中双击【Internet 协议版本 4 (TCP/Ipv4)】选项，然后在弹出的属性对话框中设置静态 IP 地址。如下图 1-11 所示。



图 1-11 修改电脑 IP 地址

### 1.3.4 绑定 ARP

本工程不支持 ARP 协议，只能通过静态绑定的方式来强制将开发板的 IP 地址和 MAC 地址关联在一起。这样，当 PC 发送给 192.168.0.2 的数据包的时候，目标 MAC 地址自动为开发板的 MAC 地址。

关于 ARP 的绑定请查看以下帖子内容：

[以太网通信静态 ARP 绑定方法与常见问题解决方案](#)

<http://www.corecourse.cn/forum.php?mod=viewthread&tid=28645>

### 1.3.5 功能验证

硬件连接后，烧录编译好的程序。

可以通过信号发生器给 ACM2108/108 模块的 ADC 通道输入一个信号源，也可以通过 ACM2108/108 自带的 DAC 功能为 ADC 提供信号源。然后通过网络调试助手或者我们提供的上位机软件采集数据。

#### 1.3.5.1 网络调试助手通信

首先需要打开网络调试助手发送指令去配置，按照如下所述设置各项参数。网络调试助手软件读者可以在 ACG525 开发板资料包的常用软件文件夹中找到。

1. 选择协议类型为 UDP。
2. 设置本地 IP 地址为 192.168.0.3。
3. 设置本地端口号为 6102。
4. 点击【连接】按钮以创建连接，连接上后该按钮为红色“断开”字样。
5. 连接上后，设置目标主机为 192.168.0.2，目标端口为 5000。
6. 点击“接收保存到文件”这几个字，在弹出的界面中设置文件路径、文件名称，如下图 1-12 所示。这样在数据接收完成之后会保存一个数据文件。方便后面进行分析。



图 1-12 设置保存文件

7. 发送设置中数据类型设置为 hex 格式，如下图 1-13 所示。



图 1-13 设置发送格式为 hex 码格式

8. 发送命令帧。

在前面接收转命令模块中介绍到数据帧格式对 ACM2108/108 模块的四个寄存器的配置。

店铺：<https://xiaomeige.taobao.com>

技术博客：<http://www.cnblogs.com/xiaomeige/>

官方网站：[www.corecourse.cn](http://www.corecourse.cn)

技术群组：

例如，PC 端要设置采样数据个数(DataNum 寄存器，地址为 2)为 16384(0x4000)个，发送数据帧内容：0x55 0xA5 0x02 0x00 0x00 0x40 0x00 0xF0。PC 端要设置采样速率(ADC\_Speed\_Set 寄存器，地址为 3)为 50M，则需要发送的数据帧内容为：0x55 0xA5 0x03 0x00 0x00 0x00 0x00 0xF0。

当上述设置都设置完成后，就可以向 0 号寄存器写入任意值，来开始一次采样传输了。数据帧内容可以为 0x55 0xA5 0x00 0x00 0x00 0x00 0x00 0xF0，这里需要注意的是 0 号寄存器必须放在最后，因为 0 号寄存器负责启动 ADC，ADC 在未配置完全的情况下开始启动，数据很容易输出错误值。

开始传输数据帧命令发送完成之后，ACM2108/108 模块就能实现以 50M 的采样速率，对 1 个通道进行采样（本次实验以通道 1 为例），共采集 16384 个数据。四个寄存器对应的配置如下表 1-10 所示：

表 1-10 ACM2108/108 数据帧格式配置表

| 寄存器名称         | 数据帧数据                   |
|---------------|-------------------------|
| DataNum       | 55 A5 02 00 00 40 00 F0 |
| ChannelSel    | 55 A5 01 00 00 00 01 F0 |
| ADC_Speed_Set | 55 A5 03 00 00 00 00 F0 |
| RestartReq    | 55 A5 00 00 00 00 00 F0 |

配置成网络调试助手发送的数据格式如下：

55A50200004000F055A50100000001F055A50300000000F055A500000000000F0

最终的网络助手配置界面如下图 1-14 所示：



图 1-14 网络调试助手配置界面

点击发送之后可以看到网络调试助手在不断的接收数据，如下图 1-15 所示。从图中可以看出一共接收到 32768 个数据，这是因为设置的 ADC 采样数量为 16384，ADC 采样数据是 16 位的，以太网是以字节（8 位）为单位进行发送的，所以通过以太网接收到字节数应该是  $16384 \times 2$  个数据，这也就是说明接收到的数据的个数没错。



图 1-15 网络调试助手接收数据

### 1.3.5.2 MATLAB 图像绘制

前面通过网络调试助手得到了 ADC 采集到的数据文件，然后我们就需要对采集到的数据进行分析，本次实验使用 MATLAB 软件进行分析。使用 MATLAB 软件需要读者电脑安装了 MTALAB，如果已经安装好了 MTALAB 软件，则可以双击我们提供的 ADCdata\_to\_wave\_v2\_2.m 文件，在打开方式里选择以 MATLAB 打开，该文件位于本次实验的工程压缩包下，将压缩包解压便可以看到该文件。文件打开之后，读者需要将代码中文件路径修改为你保存的数据文件路径，随后点击运行便可以直观的看到数据是否正确，MATLAB 操作如下图 1-16 所示，得到的波形图如下图 1-17 所示。



```

1 %=====
2 %设置ADC采集数据有效位宽
3 ADC_DATA_WIDTH = 8;
4 %ADC采集电压范围±VOL_RANGE (V)
5 VOL_RANGE = 5;
6 %采样率(Hz)
7 SAMPLE_RATE = 50000000;
8 %ADC采集电压分辨率 LSB
9 LSB = VOL_RANGE*2/(2^ADC_DATA_WIDTH);
10 %=====
11
12 %读取文件数据, 转换成波形显示
13 fileID = fopen('F:\ZYNN_Project\adc_data.dat') %在这里修改需要分
14 src_data = fread(fileID);
15 fclose(fileID);
16 src_data_hex = src_data;
17

```

图 1-16 修改文件路径并运行



图 1-17 MATLAB 分析波形图

前面我们提到过本次实验提供的信号源为 100Khz, Vpp 为 5V 的正弦波, 与 MATLAB 分析出来的波形一致, 说明我们本次实验成功。

### 1.3.5.3 数据采集上位机通信

前面通过网络调试助手采集数据时，每次保存数据都需要重新点击“接收保存文件”一栏，修改寄存器参数的时候，都需要重新计算，然后发送命令，修改之后也不能直接实时观察到数据波形，使用起来不是很方便。基于上述问题，我们设计了上位机软件“小梅哥控制台 For ADC 采集”进行数据采集，上位机内部直接对命令进行了构建，用户只需要在界面上对采样参数进行设置，便可以实时观测到数据变化，该软件的最新下载链接如下所示：

#### 数据采集上位机使用方法说明

<http://www.corecourse.cn/forum.php?mod=viewthread&tid=29224>

双击上位机软件，初始界面如下图 1-18 所示。



图 1-18 上位机软件初始界面显示

本次实验使用该软件的方式如下所示：

1. 点击 ADC，选择 ACM2108/ACM108。
2. 点击方式，选择网口，可以看到主机 IP（PC 端）和目的 IP（FPGA）以及对应的端口号。主机 IP：192.168.0.2，主机端口号：6102；目的 IP：192.168.0.3，目的端口号：5000。
3. 选择完成之后，我们可以看到采样通道、采样数量等都已经设置了初始值（默认设置的采样率为 ADC 模块的最大采样率），用户可以根据自己的需求进行修改。
4. 点击网络连接。

5. 点击开始传输之后，可以看到在右边采样电压波形图界面可以直观看到波形图，如下图 1-19 所示。需要注意的是波形图的横坐标对应的不是频率，而是采样数量。



图 1-19 数据采集上位机显示图

通过上位机采集到的数据文件保存在 d:/ MFC\_Data 文件夹下，后续可以通过 MATLAB 软件进行进一步的分析，通过 MATLAB 分析的波形图如下图 1-20 所示。从图中可以看出，采集到的数据是频率为 100Khz，电压在正负 2.5V 左右的正弦波，与我们输入的信号一致。



图 1-20 MATLAB 进一步波形分析图

### 1.3.6 ACM2108 模块\ACM108 模块产生信号源

在我们提供的例程中，包含 DDS 模块，用户可以用杜邦线直接将 ACM2108 模块的 ADC 和 DAC 相连，进行数据采集，其中按键 S2 可以修改 DAC 的信号输出。连接方式如下图 1-21 所示，ACM108 模块的连接方式如下图 1-22 所示，将 ADC0 与 DAC0 相连，这样我们就可以通过 DAC0 输出信号给 ADC0 采集，注意不要连接到 GND。



图 1-21 ACM2108 模块 ADC 和 DAC 连接示意图



图 1-22 ACM108 模块 ADC 和 DAC 连接示意图

通过按下 S0 按键切换信号源，通过示波器观测采集到的信号源，如下图 1-23 所示为 DAC 产生的 125Khz 的正弦波信号。然后通过网口助手采集数据，MATLAB 分析数据得到波形图如下图 1-24 所示。



图 1-23 DAC 输出的信号



图 1-24 ADC 采集到的信号

当通过 MATLAB 观测到的信号不明显时，可以通过按键 S0 切换到更高频率的波形进行采集。按键切换次数与对应的波形的说明如下表 1-11 所示。

表 1-11 按键切换对应波形表

| 按键次数 | 对应波形      | 按键次数 | 对应波形      | 按键次数 | 对应波形      |
|------|-----------|------|-----------|------|-----------|
| 0    | 1.25K 正弦波 | 9    | 125K 正弦波  | 18   | 6.25M 正弦波 |
| 1    | 1.25K 方波  | 10   | 125K 方波   | 19   | 6.25M 方波  |
| 2    | 1.25K 三角波 | 11   | 125K 三角波  | 20   | 6.25M 三角波 |
| 3    | 12.5K 正弦波 | 12   | 625K 正弦波  | 21   | 12.5M 正弦波 |
| 4    | 12.5K 方波  | 13   | 625K 方波   | 22   | 12.5M 方波  |
| 5    | 12.5K 三角波 | 14   | 625K 三角波  | 23   | 12.5M 三角波 |
| 6    | 62.5K 正弦波 | 15   | 1.25M 正弦波 |      |           |

|   |           |    |           |  |  |
|---|-----------|----|-----------|--|--|
| 7 | 62.5K 方波  | 16 | 1.25M 方波  |  |  |
| 8 | 62.5K 三角波 | 17 | 1.25M 三角波 |  |  |

## 1.4 思考与总结

本次实验介绍了基于 ACM2108/108 的千兆以太网收发，用户通过网口调试助手以太网帧向开发板发送指令数据配置 ACM2108/108 的四个寄存器，以此控制 ADC 进行采样，并将数据缓存后再组成以太网帧，经由网口发送至电脑，借由网口调试工具对数据进行查看。如果使用我们提供的上位机软件，则不需要自己设置命令，只需要在界面上修改相关参数，便可以在右边的波形显示界面实时观察到波形变化。