# 实验 07 FPGA 实验平台及 IP 核使用

## 简介



前面的实验中,我们完成了一个简单的电路设计,并顺利的烧写到 FPGA 实验平台上。但可能会有部分读者对于其中的一些步骤并不了解,本次实验中,我们将对所使用的 FPGA 实验平台进行介绍,以帮助读者加深对 FPGA 开发流程各环节的理解。此外,我们还会介绍到如何使用 IP 核进行电路设计。

### 实验目的

学会查看原理图

理解 FPGA 开发各关键环节

学会使用 IP核(知识产权核)

#### 实验环境

PC 一台

Windows 或 Linux 操作系统

Vivado

FPGA 实验平台 (Nexys4 DDR)

Logisim

vlab. ustc. edu. cn

#### 实验步骤

#### Step1. Nexys4 DDR 开发板简介

我们对实验中使用的 Nexys4 DDR 开发板进行简单的介绍。



Nexys4 DDR 开发板采用 Xilinx 的 XC7A100TCSG324C-1 为核心芯片,配备了开关、按键、LED 灯、数码管等多种基本外设,还支持 USB、RJ45、VGA 等多种通用接口,以及五个用于扩展的 PMOD 接口,此外还放置了一颗 128MB 的 DDR 内存颗粒。用户不仅可以在该开发板上完成各种数字逻辑实验,还可以搭建一个完整的片上系统(SOC),运行操作系统,进行单片机或嵌入式系统的开发,功能非常强大。

下图为 Nexys4 开发板支持的外设列表。本系列实验课程中,主要用到开关、按键、LED、数码管、PMOD 等基本外设。在前面的实验中,我们实现了用开关控制 LED 灯的逻辑电路,下面我们将通过开发板的原理图讲解这几种外设的工作原理和连接关系,使读者能熟练的使用它们设计出各种功能电路。

| Callout | Component Description                    | Callout | Component Description                |
|---------|------------------------------------------|---------|--------------------------------------|
| 1       | Power select jumper and battery header   | 13      | FPGA configuration reset button      |
| 2       | Shared UART/ JTAG USB port               | 14      | CPU reset button (for soft cores)    |
| 3       | External configuration jumper (SD / USB) | 15      | Analog signal Pmod connector (XADC)  |
| 4       | Pmod connector(s)                        | 16      | Programming mode jumper              |
| 5       | Microphone                               | 17      | Audio connector                      |
| 6       | Power supply test point(s)               | 18      | VGA connector                        |
| 7       | LEDs (16)                                | 19      | FPGA programming done LED            |
| 8       | Slide switches                           | 20      | Ethernet connector                   |
| 9       | Eight digit 7-seg display                | 21      | USB host connector                   |
| 10      | JTAG port for (optional) external cable  | 22      | PIC24 programming port (factory use) |
| 11      | Five pushbuttons                         | 23      | Power switch                         |
| 12      | Temperature sensor                       | 24      | Power jack                           |

Step2. 开发板原理图介绍

Nexys4 DDR 开发板上通过丝印的形式标注了部分外设与 FPGA 管脚之间的连接方式,如下图所示,五个按键的下方都注明了对应的 FPGA 管脚。这种方式对于比较简单的外设比较使用,一旦引脚数量过多,就不适用了,另外也无法得知更具体的电路连接关系(如按键按下后输出高电平还是低电平)。



虽然 FPGA 的通用管脚既可以设置为输入,也可以设置为输出,但是在实际使用时,我们会根据其与外部电路的连接关系,固定的用来做输入或输入,例如与按键、开关相连的引脚用作输入,与 LED、数码管相连的引脚用作输出。

了解开发板结构、连接关系以及其它技术细节的最有效的途径是查看说明文档和电路原理图。下图是 Nexys4DDR 开发板说明文档中给出的 FPGA 与各种基本外设的连接关系。



每个拨动开关与中学里讲到的单刀双置开关本质相同,有三个引脚,当开关推上去后,开关的中间引脚与电源相连,FPGA 管脚呈高电平,当开关拉下来后,开关的中间引脚与地相连,FPGA 管脚呈低电平。

五个按键通过下拉电阻与 FPGA 的五个通用管脚相连,当按键没有被按下时,FPGA 管脚通过下拉电阻与地相连,管脚呈低电平状态,当按键被按下后,接通高电平,管脚呈高电平状态。

Nexys4DDR开发板上有16个LED,每个LED都是P极接FPGA管脚, N极通过限流电阻接地。当FPGA管脚输出高电平时,与其连接的LED 被点亮,当 FPGA 管脚输出低电平时,LED 熄灭。

在 4 种基本外设中,数码管电路最为复杂。开发板上使用的是公阳极数码管,FPGA 管脚通过三极管放大电路控制数码管的位选择信号,每个管脚对应一个数码管,通过分析电路可以发现,当对应管脚为低电平时,三极管导通,数码管位选择信号有效。数码管的段选择信号是共用的,每个段选择信号都同时控制 8 个数码管对应的段,低电平有效。经分析可知,数码管的位选和段选信号均为低电平有效,同时为零时对应数码管的相应段点亮,否则不亮。



图中还有两个三色 LED,根据连接图可以看出,对应的 FPGA 管脚高电平有效。

下图是 Nexys4 DDR 开发板的部分原理图,包含了与 FPGA 芯片和基本外设相关的部分。通过查阅原理图,我们不仅可以了解到 FPGA 芯片与各种外设之间的连接关系,还能了解到一些更加细节的地方,







查看原理图时有几点需要特别注意,首先对于引脚数量较多的芯片,在画原理图时会按功能将其分成多个部分,以便于查看,如上图中的 FPGA 芯片,按供电、配置、通用 IO 等功能分成了 6 个部分。第二,简单的引脚连接关系可以通过连线表示,稍复杂一些的可以通过标签表示,如上面的原理图所示,其中一个拨动开关左侧标注了"SWO"的标签,FPGA 的 J15 管脚上(IC8B 最下方)也有同样的标签,说明该开关与 FPGA 的 J15 管脚是连接在一起的。

在上次的实验中,我们还用到了时钟信号,在 XDC 文件中可以看到,该信号被分配到了 E3 管脚,为什么要分配到这个管脚呢?我们通过查看原理图可以发现该 FPGA 管脚是与一个时钟芯片相连接的,该时钟芯片正常工作时能够提供一个频率为 100MHz 的时钟信号。



通过查阅原理图我们可以发现,开发板上的 FPGA 芯片的各个引脚都已经与确定的外设连接在一起,我们在使用 Vivado 进行开发时,电路模块分配应该与开发板一致。例如一个输出信号,可以将其分配给 LED 或者数码管,而不应该分配给按键,虽然这样做在 vivado 综合室不会有任何警告,但实际烧写到开发板上是无法正常工作的。为避免每次建立工程时都要通过原理图查看管脚对应关系,我们可以使用官方提供的 Nexys4DDR 的 XDC 文件,里面给出了板上 FPGA 芯片所有的引脚对应关系,用户使用时只需修改 XDC 文件内对应的信号名与自己模块内的端口名称一致即可。

Step3. FPGAOL 平台介绍



我们搭建了一套包含 20 个节点的远程 FPGA 实验平台,其结构如上图所示。每个节点包含一个 Nexys4DDR 开发板和一个树莓派单板,两者使用排线相连。树莓派对用户来说完全透明,负责与服务器端进行通信,完成烧录 FPGA,对 FPGA 管脚信号进行实时采样等功能。

用户通过网页登录系统,申请获取资源,获得设备节点后便可以 上传并烧录自己的 bit 文件,并进行一定的交互性操作。



该平台包含开关、LED、七段数码管、十六进制动态扫描数码管等多种外设。此外,用户还可以查看各端口信号的采样波形。该平台操作简单,界面直观,用户不需要购置硬件设备便可以完成 FPGA 开发流程的学习。

#### Step4. 使用时钟管理单元 IP 核

在 FPGA 开发中,有很多常用功能的模块是不需要自己开发的,用户可以复用第三方开发好的模块,这种模块被称为 IP 核。此处我们先学习时钟管理单元 IP 核的使用。

前面已经讲到,Nexys4DDR 开发板的 FPGA 芯片 E3 管脚连接了一个 100MHz 频率的时钟晶振,可用作时序逻辑电路的时钟信号。如果我们需要一个其它频率的时钟信号,例如 10MHz,应该怎么办呢?一般的做法是通过计数器产生一个低频的脉冲信号,然后再将该脉冲信号控制其他逻辑的控制信号,如下代码所示,通过 pulse\_10mhz 信号控制 1ed 信号。

```
module ttt(
input clk, rst,
output reg led);
reg [3:0] cnt;
wire pulse_10mhz;
always@(posedge clk)
begin
   if(rst)
      cnt <= 4'h0;
else if(cnt>=9)
```

其仿真波形如下图所示



有些读者喜欢将分频信号直接作为时钟使用,虽然这种设计也能工作,但我们强烈反对这种实现方式,因为这样做会产生许多无谓的警告,也容易引起电路工作的不稳定。希望读者时刻牢记,在实际的数字电路中(仿真代码除外),时钟信号是非常特殊的一种信号,它不应该出现在过程语句和连续赋值语句内部,如下所示的代码都是不推荐的(clk 为时钟信号 signal x 为其它普通信号)

```
错误示例: assign signal_1 = func(clk, xxx);
错误示例: always@(posedge signal_2) //signal_2 由 clk 生成
错误示例: always@(posedge clk or negedge clk) //两边沿不能同时使用
时钟信号只应该出现在 always 语句的时序控制部分
正确示例: always@(posedge clk) //时钟上升沿触发,同步复位或无复位
正确示例: always@(negedge clk) //时钟下降沿触发,同步复位或无复位
正确示例: always@(posedge clk or posedge rst) //异步复位,高有效
正确示例: always@(posedge clk or negedge rst n) //异步复位,低有效
```

如果设计中确实需要不同频率的时钟信号应该通过时钟管理单元 IP 核生成。首先,点击"IP catalog",在对应窗口中选中"Clocking

Wizard"并双击。



在弹出的窗口中,对其进行设置,输入时钟频率设置为100MHz,输出时钟有两个,分别为10MHz和200MHz。





设置完成后点击确认按钮,生成 IP 核。生成的 IP 文件可在"工程目录/工程名.srcs/sources\_1/ip/IP 核名称/IP 核名称.v"找到。用户可在设计文件中像调用其它模块一样使用该 IP 核,使用时只需要了解 IP 核的功能及端口信号的含义及时序,而不用关心模块内部的具体实现。如下代码所示,分别同 10M 和 200M 的时钟作为计数器的计数时钟,通过仿真和烧写 FPGA,可以发现两个时钟频率的明显差异。

```
module test(
input clk,
input rst,
output [7:0] led);
wire clk_10m, clk_200m, locked;
reg [31:0] cnt_1, cnt_2;
always@(posedge clk_200m)
begin
```

```
if (~locked)
        cnt_1 <= 32'hAAAA_AAAA;
    else
        cnt 1 <= cnt 1+1'b1;;
end
always@(posedge clk_10m)
begin
    if (~locked)
        cnt 2 <= 32'hAAAA AAAA;
    else
        cnt_2 <= cnt_2+1'b1;;
end
assign led = {cnt_1[27:24], cnt_2[27:24]};
clk wiz 0 clk wiz 0 inst(
.clk\_in1
            (c1k),
.clk_out1 (clk_10m),
.c1k_out2 (c1k_200m),
. reset
          (rst).
. locked
            (locked));
endmodule
```

强烈建议读者在综合烧写之前先进行功能仿真,以确保电路功能

的正确性,下面是一段最简单的仿真测试文件。

```
`timescale Ins / 1ps
module tb( );
reg clk, rst;
initial
begin
    c1k = 0;
    forever
    #5 c1k = c1k;
end
initial
begin
    rst = 1;
    #100 rst = 0;
end
        test (
test
.c1k
        (c1k),
.rst
        (rst),
        ());
. 1ed
endmodule
```

其仿真波形如下所示:



对于 Nexys4DDR 开发板,我们将 LED 信号分配到最右侧的 8 个 LED 灯上,其 XDC 文件如下:

```
set_property -dict { PACKAGE_PIN E3
                                        IOSTANDARD LVCMOS33 } [get_ports { clk }];
                                        IOSTANDARD LVCMOS33 } [get_ports { rst }];
set property -dict { PACKAGE PIN N17
## leds
set_property -dict { PACKAGE_PIN H17
                                       IOSTANDARD LVCMOS33 } [get_ports { led[0] }];
                                       IOSTANDARD LVCMOS33 } [get ports { led[1] }];
set property -dict { PACKAGE PIN K15
set_property -dict { PACKAGE_PIN J13
                                       IOSTANDARD LVCMOS33 } [get_ports { led[2] }];
set property-dict { PACKAGE PIN N14
                                       IOSTANDARD LVCMOS33 } [get ports { led[3] }];
set property -dict { PACKAGE PIN R18
                                       IOSTANDARD LVCMOS33 } [get_ports { led[4] }];
set property -dict { PACKAGE PIN V17
                                       IOSTANDARD LVCMOS33 } [get ports { led[5] }];
set property -dict { PACKAGE PIN U17
                                       IOSTANDARD LVCMOS33 } [get_ports { led[6] }];
set_property -dict { PACKAGE_PIN U16
                                       IOSTANDARD LVCMOS33 } [get ports { led[7] }];
```

需要特别指出的是,时钟管理单元并不能产生任意频率的时钟信号,以前面为例,输入时钟为 100MHz 时,输出时钟只能是频率范围在 4.687MHz~800MHz 之间的某些频率,如 4MHz 超出范围无法产生,799MHz 虽然在频率范围内,但因精度问题,只能产生以及频率相近的时钟信号,有兴趣的读者可自行验证。

如果电路中需要一个较低频率的时序,例如我们需要一个每秒钟加一的计数器,那我们就只能通过前面介绍的方法,用周期为1秒的脉冲信号来控制了。

#### Step5. 使用片内存储单元

在前面的实验中,我们已经对存储器的行为特性进行了简单的介绍,例如 ROM 为只读存储器,包含了地址和数据两组端口,ROM 内部包含 2 <sup>地址位宽</sup>个存储单元,每个单元里存储了与数据端口相同位宽的数

据,存储器电路会将与地址相对应单元的数据呈现在数据端口上,下 图是一个包含 2<sup>2</sup>=4 个存储单元的 ROM,数据位宽为 16 位。



我们可以从 ROM 读取数据,但无法通过端口修改其内容,与此相对应的是 RAM (随机存储器),其内容可读可写,下图为 Logisim 中的 RAM模型,包含地址(A)、数据(D,输入输入复用)、片选(sel)、时钟(clk,数据可在时钟上升沿写入)、输出使能(ld,为1 时数据端口为输出,否则为输入)、清空(clr)等端口信号。这种 RAM包含一套读写端口,因此成为单端口 RAM。读者可在 Logisim 中实际体验各端口信号的功能。



Vivao中也提供了存储器相关的 IP 核,种类和接口类型比 Logisim中更加丰富,我们通过实际的例子来学习一下。 IP 核目录中提供了两种存储器实现方式: "Distributed Memory"和"Block Memory"两种,我们以"Distributed Memory"为例进行介绍。



在存储器参数设置页面,用户可以对 IP 核名称、存储单元深度和

位宽、存储器类型等进行设置。在页面的左侧是模块的端口图。在上图中,存储器深度设为 16=2<sup>4</sup>,数据位宽为 3 位,因此地址信号为 4 位,数据端口位宽为 3。存储器类型选为简单双端口,因此包含了两套端口,其中 dpra、dpo 构成了读端口,d、a、clk、we 构成了写端口。在"RST&Initialization"页面,我们还可以通过后缀为 coe 的文件对存储器进行初始化,如下图所示,coe 文件格式非常简单,可以以文本方式打开和编辑。



#### An example COE file:

memory\_initialization\_radix = 16; memory\_initialization\_vector = 23f4 0721 11ff ABe1 0001 1 0A 0 23f4 0721 11ff ABe1 0001 1 0A 0 23f4 721 11ff ABe1 0001 1 A 0 23f4 721 11ff ABe1 0001 1 A 0;

我们可以像调用其它模块一样调用该模块,其仿真波形如下图所示,可以看到,5号地址的初始值为1,与coe文件相一致,在225ns时钟的上升沿处,通过写端口将该地址改写成了5,因此在236ns处读端口从该地读取到的是改写后的数值。



除了时钟管理单元和存储单元外,Vivado 中还提供了许多功能的 IP 核,在 IP 核参数设置界面提供了说明文档的连接,读者可选择自己感兴趣的功能模块自行学习。

#### 实验练习

**题目 1.** 例化一个 16\*8bit 的 ROM, 并对其进行初始化,输入端口由 4 个开关控制,输出端口连接到七段数码管上(可只用一个数码管显

示,也可8个数码管同时显示相同的数值),控制数码管显示与开关相对应的十六进制数字,例如四个开关输入全为零时,数码管显示"0",输入全为1时,数码管显示"F"。

**题目 2.** 采用 8 个开关作为输入,两个数码管作为输出,采用时分复用的方式将开关的十六进制数值在两个数码管上显示出来,例如高四位全为 1, 低四位全为 0 时, 数码管显示"F0"。

**题目 3.** 利用本实验中的时钟管理单元或周期脉冲技术,设计一个精度为 0.1 秒的计时器,用 4 位数码管显示出来,数码管从高到低,分别表示分钟、秒钟十位、秒钟个位、十分之一秒,该计时器具有复位功能(可采用按键或开关作为复位信号),复位时计数值为 1234,即 1 分 23.4 秒

#### 总结与思考

- 1. 请总结本次实验的收获
- 2. 请评价本次实验的难易程度
- 3. 请评价本次实验的任务量
- 4. 请为本次实验提供改进建议