ZYNQ之FPGA学习----UART串口实验

Bertha 。 2023-09-23 16:52 229阅读 0赞

1 UART串口简介

UART串口基础知识学习:硬件设计基础——通信协议UART

2 实验任务

上位机通过串口调试助手发送数据给 Zynq,Zynq PL 端通过 RS232 串口接收数据并将接收到的数据发送给上位机,完成串口数据环回,管脚分配如下:

在这里插入图片描述
图片来自《领航者ZYNQ之FPGA开发指南》

3 实验设计

3.1 创建工程

新建工程,操作如图所示:

在这里插入图片描述

输入工程名字和工程路径,如图:

在这里插入图片描述

选择创建RTL工程,如图:

在这里插入图片描述

直接点击Next:

在这里插入图片描述

继续点击Next:

在这里插入图片描述

添加芯片型号,操作如图:

在这里插入图片描述

完成工程创建:

在这里插入图片描述

3.2 设计输入

创建工程顶层文件,操作如图:

在这里插入图片描述

创建uart_loopback_top顶层模块:

在这里插入图片描述

创建完成:

在这里插入图片描述

双击打开,输入代码如下:

  1. module uart_loopback_top(
  2. input sys_clk, //外部 50M 时钟
  3. input sys_rst_n, //外部复位信号,低有效
  4. input uart_rxd, //UART 接收端口
  5. output uart_txd //UART 发送端口
  6. );
  7. //parameter define
  8. parameter CLK_FREQ = 50000000; //定义系统时钟频率
  9. parameter UART_BPS = 115200; //定义串口波特率
  10. //wire define
  11. wire uart_recv_done; //UART 接收完成
  12. wire [7:0] uart_recv_data; //UART 接收数据
  13. wire uart_send_en; //UART 发送使能
  14. wire [7:0] uart_send_data; //UART 发送数据
  15. wire uart_tx_busy; //UART 发送忙状态标志
  16. //串口接收模块
  17. uart_recv #(
  18. .CLK_FREQ (CLK_FREQ ), //设置系统时钟频率
  19. .UART_BPS (UART_BPS )) //设置串口接收波特率
  20. u_uart_recv(
  21. .sys_clk (sys_clk ),
  22. .sys_rst_n (sys_rst_n ),
  23. .uart_rxd (uart_rxd ),
  24. .uart_done (uart_recv_done ),
  25. .uart_data (uart_recv_data)
  26. );
  27. //串口发送模块
  28. uart_send #(
  29. .CLK_FREQ (CLK_FREQ ), //设置系统时钟频率
  30. .UART_BPS (UART_BPS )) //设置串口发送波特率
  31. u_uart_send(
  32. .sys_clk (sys_clk ),
  33. .sys_rst_n (sys_rst_n ),
  34. .uart_en (uart_send_en ),
  35. .uart_din (uart_send_data ),
  36. .uart_tx_busy (uart_tx_busy ),
  37. .uart_txd (uart_txd)
  38. );
  39. //串口环回模块
  40. uart_loop u_uart_loop(
  41. .sys_clk (sys_clk ),
  42. .sys_rst_n (sys_rst_n ),
  43. .recv_done (uart_recv_done ), //接收一帧数据完成标志信号
  44. .recv_data (uart_recv_data ), //接收的数据
  45. .tx_busy (uart_tx_busy ), //发送忙状态标志
  46. .send_en (uart_send_en ), //发送使能信号
  47. .send_data (uart_send_data) //待发送数据
  48. );
  49. endmodule

如图所示:

在这里插入图片描述

创建uart_recv文件,如图:

在这里插入图片描述

双击打开,输入代码:

  1. module uart_recv(
  2. input sys_clk, //系统时钟
  3. input sys_rst_n, //系统复位,低电平有效
  4. input uart_rxd, //UART 接收端口
  5. output reg uart_done, //接收一帧数据完成标志
  6. output reg [7:0] uart_data //接收的数据
  7. );
  8. //parameter define
  9. parameter CLK_FREQ = 50000000; //系统时钟频率
  10. parameter UART_BPS = 9600; //串口波特率
  11. localparam BPS_CNT = CLK_FREQ/UART_BPS; //为得到指定波特率,
  12. //需要对系统时钟计数 BPS_CNT 次
  13. //reg define
  14. reg uart_rxd_d0;
  15. reg uart_rxd_d1;
  16. reg [15:0] clk_cnt; //系统时钟计数器
  17. reg [ 3:0] rx_cnt; //接收数据计数器
  18. reg rx_flag; //接收过程标志信号
  19. reg [ 7:0] rxdata; //接收数据寄存器
  20. //wire define
  21. wire start_flag;
  22. //捕获接收端口下降沿(起始位),得到一个时钟周期的脉冲信号
  23. assign start_flag = uart_rxd_d1 & (~uart_rxd_d0 );
  24. //对 UART 接收端口的数据延迟两个时钟周期
  25. always @( posedge sys_clk or negedge sys_rst_n) begin
  26. if (!sys_rst_n) begin
  27. uart_rxd_d0 <= 1'b0;
  28. uart_rxd_d1 <= 1'b0;
  29. end
  30. else begin
  31. uart_rxd_d0 <= uart_rxd;
  32. uart_rxd_d1 <= uart_rxd_d0;
  33. end
  34. end
  35. //当脉冲信号 start_flag 到达时,进入接收过程
  36. always @( posedge sys_clk or negedge sys_rst_n) begin
  37. if (!sys_rst_n)
  38. rx_flag <= 1'b0;
  39. else begin
  40. if(start_flag) //检测到起始位
  41. rx_flag <= 1'b1; //进入接收过程,标志位 rx_flag 拉高
  42. //计数到停止位中间时,停止接收过程
  43. else if ((rx_cnt == 4'd9) && (clk_cnt == BPS_CNT/2 ))
  44. rx_flag <= 1'b0; //接收过程结束,标志位 rx_flag 拉低
  45. else
  46. rx_flag <= rx_flag;
  47. end
  48. end
  49. //进入接收过程后,启动系统时钟计数器
  50. always @( posedge sys_clk or negedge sys_rst_n) begin
  51. if (!sys_rst_n)
  52. clk_cnt <= 16'd0;
  53. else if ( rx_flag ) begin //处于接收过程
  54. if (clk_cnt < BPS_CNT - 1)
  55. clk_cnt <= clk_cnt + 1'b1;
  56. else
  57. clk_cnt <= 16'd0; //对系统时钟计数达一个波特率周期后清零
  58. end
  59. else
  60. clk_cnt <= 16'd0; //接收过程结束,计数器清零
  61. end
  62. //进入接收过程后,启动接收数据计数器
  63. always @( posedge sys_clk or negedge sys_rst_n) begin
  64. if (!sys_rst_n)
  65. rx_cnt <= 4'd0;
  66. else if ( rx_flag ) begin //处于接收过程
  67. if (clk_cnt == BPS_CNT - 1) //对系统时钟计数达一个波特率周期
  68. rx_cnt <= rx_cnt + 1'b1; //此时接收数据计数器加 1
  69. else
  70. rx_cnt <= rx_cnt;
  71. end
  72. else
  73. rx_cnt <= 4'd0; //接收过程结束,计数器清零
  74. end
  75. //根据接收数据计数器来寄存 uart 接收端口数据
  76. always @( posedge sys_clk or negedge sys_rst_n) begin
  77. if ( !sys_rst_n)
  78. rxdata <= 8'd0;
  79. else if(rx_flag) //系统处于接收过程
  80. if (clk_cnt == BPS_CNT/2) begin //判断系统时钟计数器计数到数据位中间
  81. case ( rx_cnt )
  82. 4'd1 : rxdata[0] <= uart_rxd_d1; //寄存数据位最低位
  83. 4'd2 : rxdata[1] <= uart_rxd_d1;
  84. 4'd3 : rxdata[2] <= uart_rxd_d1;
  85. 4'd4 : rxdata[3] <= uart_rxd_d1;
  86. 4'd5 : rxdata[4] <= uart_rxd_d1;
  87. 4'd6 : rxdata[5] <= uart_rxd_d1;
  88. 4'd7 : rxdata[6] <= uart_rxd_d1;
  89. 4'd8 : rxdata[7] <= uart_rxd_d1; //寄存数据位最高位
  90. default :;
  91. endcase
  92. end
  93. else
  94. rxdata <= rxdata;
  95. else
  96. rxdata <= 8'd0;
  97. end
  98. //数据接收完毕后给出标志信号并寄存输出接收到的数据
  99. always @( posedge sys_clk or negedge sys_rst_n) begin
  100. if (!sys_rst_n) begin
  101. uart_data <= 8'd0;
  102. uart_done <= 1'b0;
  103. end
  104. else if(rx_cnt == 4'd9) begin //接收数据计数器计数到停止位时
  105. uart_data <= rxdata; //寄存输出接收到的数据
  106. uart_done <= 1'b1; //并将接收完成标志位拉高
  107. end
  108. else begin
  109. uart_data <= 8'd0;
  110. uart_done <= 1'b0;
  111. end
  112. end
  113. endmodule

如图所示:

在这里插入图片描述

创建uart_send文件,如图所示:

在这里插入图片描述

双击打开文件,输入代码:

  1. module uart_send(
  2. input sys_clk, //系统时钟
  3. input sys_rst_n, //系统复位,低电平有效
  4. input uart_en, //发送使能信号
  5. input [7:0] uart_din, //待发送数据
  6. output uart_tx_busy, //发送忙状态标志
  7. output reg uart_txd //UART 发送端口
  8. );
  9. //parameter define
  10. parameter CLK_FREQ = 50000000; //系统时钟频率
  11. parameter UART_BPS = 9600; //串口波特率
  12. localparam BPS_CNT = CLK_FREQ/UART_BPS; //为得到指定波特率,对系统时钟计数 BPS_CNT 次
  13. //reg define
  14. reg uart_en_d0;
  15. reg uart_en_d1;
  16. reg [15:0] clk_cnt; //系统时钟计数器
  17. reg [ 3:0] tx_cnt; //发送数据计数器
  18. reg tx_flag; //发送过程标志信号
  19. reg [ 7:0] tx_data; //寄存发送数据
  20. //wire define
  21. wire en_flag;
  22. assign uart_tx_busy = tx_flag;
  23. //捕获 uart_en 上升沿,得到一个时钟周期的脉冲信号
  24. assign en_flag = (~uart_en_d1) & uart_en_d0;
  25. //对发送使能信号 uart_en 延迟两个时钟周期
  26. always @( posedge sys_clk or negedge sys_rst_n) begin
  27. if (!sys_rst_n) begin
  28. uart_en_d0 <= 1'b0;
  29. uart_en_d1 <= 1'b0;
  30. end
  31. else begin
  32. uart_en_d0 <= uart_en;
  33. uart_en_d1 <= uart_en_d0;
  34. end
  35. end
  36. //当脉冲信号 en_flag 到达时,寄存待发送的数据,并进入发送过程
  37. always @( posedge sys_clk or negedge sys_rst_n) begin
  38. if (!sys_rst_n) begin
  39. tx_flag <= 1'b0;
  40. tx_data <= 8'd0;
  41. end
  42. else if (en_flag) begin //检测到发送使能上升沿
  43. tx_flag <= 1'b1; //进入发送过程,标志位 tx_flag 拉高
  44. tx_data <= uart_din; //寄存待发送的数据
  45. end
  46. //计数到停止位结束时,停止发送过程
  47. else if ((tx_cnt == 4'd9) && (clk_cnt == BPS_CNT -(BPS_CNT/16) )) begin
  48. tx_flag <= 1'b0; //发送过程结束,标志位 tx_flag 拉低
  49. tx_data <= 8'd0;
  50. end
  51. else begin
  52. tx_flag <= tx_flag;
  53. tx_data <= tx_data;
  54. end
  55. end
  56. //进入发送过程后,启动系统时钟计数器
  57. always @( posedge sys_clk or negedge sys_rst_n) begin
  58. if (!sys_rst_n)
  59. clk_cnt <= 16'd0;
  60. else if (tx_flag) begin //处于发送过程
  61. if (clk_cnt < BPS_CNT - 1)
  62. clk_cnt <= clk_cnt + 1'b1;
  63. else
  64. clk_cnt <= 16'd0; //对系统时钟计数达一个波特率周期后清零
  65. end
  66. else
  67. clk_cnt <= 16'd0; //发送过程结束
  68. end
  69. //进入发送过程后,启动发送数据计数器
  70. always @( posedge sys_clk or negedge sys_rst_n) begin
  71. if (!sys_rst_n)
  72. tx_cnt <= 4'd0;
  73. else if (tx_flag) begin //处于发送过程
  74. if (clk_cnt == BPS_CNT - 1) //对系统时钟计数达一个波特率周期
  75. tx_cnt <= tx_cnt + 1'b1; //此时发送数据计数器加 1
  76. else
  77. tx_cnt <= tx_cnt;
  78. end
  79. else
  80. tx_cnt <= 4'd0; //发送过程结束
  81. end
  82. //根据发送数据计数器来给 uart 发送端口赋值
  83. always @( posedge sys_clk or negedge sys_rst_n) begin
  84. if (!sys_rst_n)
  85. uart_txd <= 1'b1;
  86. else if (tx_flag)
  87. case(tx_cnt)
  88. 4'd0: uart_txd <= 1'b0; //起始位
  89. 4'd1: uart_txd <= tx_data[0 ]; //数据位最低位
  90. 4'd2: uart_txd <= tx_data[1 ];
  91. 4'd3: uart_txd <= tx_data[2 ];
  92. 4'd4: uart_txd <= tx_data[3 ];
  93. 4'd5: uart_txd <= tx_data[4 ];
  94. 4'd6: uart_txd <= tx_data[5 ];
  95. 4'd7: uart_txd <= tx_data[6 ];
  96. 4'd8: uart_txd <= tx_data[7 ]; //数据位最高位
  97. 4'd9: uart_txd <= 1'b1; //停止位
  98. default: ;
  99. endcase
  100. else
  101. uart_txd <= 1'b1; //空闲时发送端口为高电平
  102. end
  103. endmodule

如图所示:

在这里插入图片描述

创建uart_loop文件,如图所示:

在这里插入图片描述

双击打开,输入代码:

  1. module uart_loop(
  2. input sys_clk, //系统时钟
  3. input sys_rst_n, //系统复位,低电平有效
  4. input recv_done, //接收一帧数据完成标志
  5. input [7:0] recv_data, //接收的数据
  6. input tx_busy, //发送忙状态标志
  7. output reg send_en, //发送使能信号
  8. output reg [7:0] send_data //待发送数据
  9. );
  10. reg recv_done_d0;
  11. reg recv_done_d1;
  12. reg tx_ready;
  13. wire recv_done_flag;
  14. //捕获 recv_done 上升沿,得到一个时钟周期的脉冲信号
  15. assign recv_done_flag = (~recv_done_d1) & recv_done_d0;
  16. //对发送使能信号 recv_done 延迟两个时钟周期
  17. always @( posedge sys_clk or negedge sys_rst_n) begin
  18. if (!sys_rst_n) begin
  19. recv_done_d0 <= 1'b0;
  20. recv_done_d1 <= 1'b0;
  21. end
  22. else begin
  23. recv_done_d0 <= recv_done;
  24. recv_done_d1 <= recv_done_d0;
  25. end
  26. end
  27. //判断接收完成信号,并在串口发送模块空闲时给出发送使能信号
  28. always @( posedge sys_clk or negedge sys_rst_n) begin
  29. if (!sys_rst_n) begin
  30. tx_ready <= 1'b0;
  31. send_en <= 1'b0;
  32. send_data <= 8'd0;
  33. end
  34. else begin
  35. if(recv_done_flag) begin //检测串口接收到数据
  36. tx_ready <= 1'b1; //准备启动发送过程
  37. send_en <= 1'b0;
  38. send_data <= recv_data; //寄存串口接收的数据
  39. end
  40. else if(tx_ready && (~tx_busy )) begin //检测串口发送模块空闲
  41. tx_ready <= 1'b0; //准备过程结束
  42. send_en <= 1'b1; //拉高发送使能信号
  43. end
  44. end
  45. end
  46. endmodule

如图所示:
在这里插入图片描述

3.3 分析与综合

对设计进行分析,操作如图:

在这里插入图片描述

分析后的设计,Vivado自动生成顶层原理图,如图:

在这里插入图片描述

对设计进行综合,操作如图:

在这里插入图片描述

综合完成后,弹出窗口如下,直接关闭:

在这里插入图片描述

3.4 约束输入

创建约束文件,操作如图所示:

在这里插入图片描述

创建约束文件,输入文件名:

在这里插入图片描述
双击打开,输入约束代码:

  1. create_clock -period 20.000 -name clk [get_ports sys_clk]
  2. set_property -dict {
  3. PACKAGE_PIN U18 IOSTANDARD LVCMOS33} [get_ports sys_clk]
  4. set_property -dict {
  5. PACKAGE_PIN J15 IOSTANDARD LVCMOS33} [get_ports sys_rst_n]
  6. set_property -dict {
  7. PACKAGE_PIN J14 IOSTANDARD LVCMOS33} [get_ports uart_rxd]
  8. set_property -dict {
  9. PACKAGE_PIN K18 IOSTANDARD LVCMOS33} [get_ports uart_txd]

如图所示:

在这里插入图片描述

3.5 设计实现

点击 Flow Navigator 窗口中的 Run Implementation,如图所示:

在这里插入图片描述

点击OK:

在这里插入图片描述

完成后,直接关闭:

在这里插入图片描述

3.6 功能仿真

创建TestBench,操作如图所示:

在这里插入图片描述

创建激励文件,输入文件名:

在这里插入图片描述

创建完成:

在这里插入图片描述

双击打开,输入TestBench(激励)代码:

  1. `timescale 1ns / 1ps
  2. module tb_uart_loopback_top();
  3. reg sys_clk;
  4. reg sys_rst_n;
  5. reg uart_rxd;
  6. wire uart_txd;
  7. parameter SYS_CLK = 50; //系统时钟频率,单位Mhz
  8. parameter SYS_PRE = 1000/(SYS_CLK*2); //时钟周期,单位ns
  9. parameter UART_BPS = 115200; //串口波特率
  10. parameter BPS_CNT = 1_000_000_000/UART_BPS;//用于串口仿真的时延
  11. //初始时刻定义
  12. initial begin
  13. $timeformat(-9, 0, " ns", 10); //定义时间显示格式
  14. sys_clk = 1'b0;
  15. sys_rst_n = 1'b0;
  16. uart_rxd <= 1'b1;
  17. #200 //系统开始工作
  18. sys_rst_n = 1'b1;
  19. #3000
  20. rx_byte({
  21. $random} % 256); //生成8位随机数1
  22. rx_byte({
  23. $random} % 256); //生成8位随机数2
  24. rx_byte({
  25. $random} % 256); //生成8位随机数3
  26. rx_byte({
  27. $random} % 256); //生成8位随机数4
  28. rx_byte({
  29. $random} % 256); //生成8位随机数5
  30. rx_byte({
  31. $random} % 256); //生成8位随机数6
  32. rx_byte({
  33. $random} % 256); //生成8位随机数7
  34. rx_byte({
  35. $random} % 256); //生成8位随机数8
  36. #600
  37. $finish();
  38. end
  39. //定义任务,每次发送的数据10 位(起始位1+数据位8+停止位1)
  40. task rx_byte(
  41. input [7:0] data
  42. );
  43. integer i; //定义一个常量
  44. //用 for 循环产生一帧数据,for 括号中最后执行的内容只能写 i=i+1
  45. for(i=0; i<10; i=i+1) begin
  46. case(i)
  47. 0: uart_rxd <= 1'b0; //起始位
  48. 1: uart_rxd <= data[0]; //LSB
  49. 2: uart_rxd <= data[1];
  50. 3: uart_rxd <= data[2];
  51. 4: uart_rxd <= data[3];
  52. 5: uart_rxd <= data[4];
  53. 6: uart_rxd <= data[5];
  54. 7: uart_rxd <= data[6];
  55. 8: uart_rxd <= data[7]; //MSB
  56. 9: uart_rxd <= 1'b1; //停止位
  57. endcase
  58. #BPS_CNT; //每发送 1 位数据延时
  59. end
  60. endtask //任务结束
  61. //设置主时钟
  62. always #10 sys_clk <= ~sys_clk; //时钟20ns,50M
  63. //例化被测试的串口接收驱动
  64. //例化被测试的串口接收驱动
  65. uart_loopback_top
  66. #(
  67. .UART_BPS (UART_BPS),
  68. .SYS_CLK (SYS_CLK)
  69. )
  70. u_uart_loopback_top(
  71. .sys_clk (sys_clk),
  72. .sys_rst_n (sys_rst_n),
  73. .uart_rxd (uart_rxd),
  74. .uart_txd (uart_txd)
  75. );
  76. endmodule

如图所示:

在这里插入图片描述

开始进行仿真,操作如下:

在这里插入图片描述

开始仿真,仿真波形如图:

在这里插入图片描述

选择需要观察的HDL仿真对象,波形如图:

在这里插入图片描述

3.7 下载验证

由于疫情,一直无法去实验室,故ZYNQ开发板不在身边,该步骤内容待更新

致谢领航者ZYNQ开发板,开启FPGA学习之路!

希望本文对大家有帮助,上文若有不妥之处,欢迎指正

分享决定高度,学习拉开差距

发表评论

表情:
评论列表 (有 0 条评论,229人围观)

还没有评论,来说两句吧...

相关阅读