当前位置:   article > 正文

UART串口通信协议

uart串口通信协议

一、串行通信

串行通信分为两种方式:同步串行通信异步串行通信

同步串行通信需要通信双方在同一时钟的控制下,同步传输数据。

异步串行通信是指通信双方使用各自的时钟控制数据的发送和接收过程。

二、UART

通用异步收发传输器(Universal Asynchronous Receiver/Transmitter,UART)是一种全双工、异步串行通信方式它在发送数据时将并行数据转换成串行数据来传输,在接收数据时将接收到的串行数
据转换成并行数据,可以实现全双工传输和接收。

UART串口通信需要两根信号线来实现,一根用于串口发送,另外一根负责串口接收

UART在发送或接收过程中的一帧数据由4部分组成,起始位(低电平)、数据位、奇偶校验位和停止位(高电平)

起始位标志着一帧数据的开始,停止位标志着一帧数据的结束,数据位是一帧数据中的有效数据。

校验位分为奇校验和偶校验,用于检验数据在传输过程中是否出错。奇校验时,发送方应使数据位中1的个数与校验位中1的个数之和为奇数;接收方在接收数据时,对1的个数进行检查,若不为奇数,则说明数据在传输过程中出了差错。同样,偶校验则检查1的个数是否为偶数。

UART通信过程中的数据格式及传输速率是可设置的,数据位可选择为 5、6、7、8位(最常用);校验位可选择奇校验、偶校验或者无校验位;停止位可选择1位(默认),1.5或2位。

串口通信的速率用波特率表示,它表示每秒传输二进制数据的位数,单位是bps(位/秒),常用的波特率有9600、19200、38400、57600以及115200等。

FPGA开发串口中,为了得到串口传输一位数据所需要的系统时钟周期数,需要对系统时钟进行计数,计数方法为FPGA时钟频率/波特率。如时钟频率为50000000hz(50MHz),需要的比特率为9600bps,则需要计数50000000/9600≈5208次

三、FPGA实现

上位机通过串口调试助手发送数据给FPGA,FPGA 通过USB串口接收数据并将接收到的数据发送给上位机,完成串口数据环回。

首先要有一个接收模块,接收串口调试助手发送的数据

  1. module uart_rx(
  2. //**************输入**************
  3. input clk, //系统50MHz时钟
  4. input rst_n, //系统复位,低电平有效
  5. input uart_rxd, //UART接收端口
  6. //**************输出**************
  7. output reg uart_done, //接收一帧数据完成标志
  8. output reg rx_flag, //接收标志位
  9. output reg [3:0] rx_cnt, //接收数据计数器
  10. output reg [7:0] rxdata, //接收端口数据寄存
  11. output reg [7:0] uart_data //接收的数据
  12. );
  13. //************参数定义************
  14. parameter CLK_FREQ = 50000000; //系统时钟
  15. parameter UART_BPS = 9600; //串口波特率
  16. localparam BPS_CNT = CLK_FREQ/UART_BPS; //波特率计数,串口传输一位所需要的系统时钟周期数
  17. //************信号定义************
  18. reg uart_rxd_1; //异步信号会带来亚稳态,常用处理方式是打拍处理,第一拍
  19. reg uart_rxd_2; //第二拍,通常打两拍就基本上就能避免亚稳态问题
  20. reg [15:0] clk_cnt; //系统时钟计数器
  21. wire start_flag; //起始标志位
  22. //检测接收端口下降沿来捕获起始位,输出一个时钟周期的脉冲start fag,并进入串口接收过程
  23. assign start_flag = uart_rxd_2 & (~uart_rxd_1);
  24. //对UART接收端口的数据延迟两个时钟周期避免亚稳态
  25. always @(posedge clk or negedge rst_n) begin
  26. if (!rst_n) begin
  27. uart_rxd_1 <= 1'b0;
  28. uart_rxd_2 <= 1'b0;
  29. end
  30. else begin
  31. uart_rxd_1 <= uart_rxd;
  32. uart_rxd_2 <= uart_rxd_1;
  33. end
  34. end
  35. //当脉冲信号start_flag有效,进入数据接收过程
  36. always @(posedge clk or negedge rst_n) begin
  37. if (!rst_n)
  38. rx_flag <= 1'b0;
  39. else begin
  40. //检测到起始位,进入数据接收过程,标志位rx_flag拉高
  41. if(start_flag)
  42. rx_flag <= 1'b1;
  43. //当接收数据计数器计数到9(9个波特周期)且在停止位中间(数据寄存已经完成,为检测下一帧数据起始位准备),停止接收,标志位rx_flag拉低
  44. else if((rx_cnt == 4'd9) && (clk_cnt == BPS_CNT/2))
  45. rx_flag <= 1'b0;
  46. else
  47. rx_flag <= rx_flag;
  48. end
  49. end
  50. //进入数据接收过程启动系统时钟计数器
  51. always @(posedge clk or negedge rst_n) begin
  52. if (!rst_n)
  53. clk_cnt <= 16'd0;
  54. //标志位rx_flag有效,处于接收过程
  55. else if (rx_flag) begin
  56. //计数没到一个波特率周期则一直计数,到了则清零
  57. if (clk_cnt < BPS_CNT - 1)
  58. clk_cnt <= clk_cnt + 1'b1;
  59. else
  60. clk_cnt <= 16'd0;
  61. end
  62. //数据接收过程结束,计数器清零
  63. else
  64. clk_cnt <= 16'd0;
  65. end
  66. //进入数据接收过程启动接收数据计数器
  67. always @(posedge clk or negedge rst_n) begin
  68. if (!rst_n)
  69. rx_cnt <= 4'd0;
  70. else if (rx_flag) begin
  71. //系统时钟计数到一个波特率周期,接收数据计数器加1,没到则不加
  72. if (clk_cnt == BPS_CNT - 1)
  73. rx_cnt <= rx_cnt + 1'b1; //可以用来判断当前传输的是第几位
  74. else
  75. rx_cnt <= rx_cnt;
  76. end
  77. //接收过程结束,计数器清零
  78. else
  79. rx_cnt <= 4'd0;
  80. end
  81. //根据接收数据计数器来寄存uart接收端口数据
  82. always @(posedge clk or negedge rst_n) begin
  83. if (!rst_n)
  84. rxdata <= 8'd0;
  85. else if(rx_flag)
  86. //计数到数据中间的采样结果最稳定
  87. if (clk_cnt == BPS_CNT/2) begin
  88. case (rx_cnt)
  89. //根据rx_cnt的值将uart接收端口的数据寄存到接收数据寄存器对应的位实现串并转换
  90. 4'd1 : rxdata[0] <= uart_rxd_2; //寄存数据位最低位
  91. 4'd2 : rxdata[1] <= uart_rxd_2;
  92. 4'd3 : rxdata[2] <= uart_rxd_2;
  93. 4'd4 : rxdata[3] <= uart_rxd_2;
  94. 4'd5 : rxdata[4] <= uart_rxd_2;
  95. 4'd6 : rxdata[5] <= uart_rxd_2;
  96. 4'd7 : rxdata[6] <= uart_rxd_2;
  97. 4'd8 : rxdata[7] <= uart_rxd_2; //寄存数据位最高位
  98. default:;
  99. endcase
  100. end
  101. else
  102. rxdata <= rxdata;
  103. else
  104. rxdata <= 8'd0;
  105. end
  106. //数据接收完毕后给出标志信号并寄存输出接收到的数据
  107. always @(posedge clk or negedge rst_n) begin
  108. if (!rst_n) begin
  109. uart_data <= 8'd0;
  110. uart_done <= 1'b0;
  111. end
  112. //接收数据计数器计数到停止位时,寄存输出接收到的数据并将接收完成标志位拉高
  113. else if(rx_cnt == 4'd9) begin
  114. uart_data <= rxdata;
  115. uart_done <= 1'b1;
  116. end
  117. else begin
  118. uart_data <= 8'd0;
  119. uart_done <= 1'b0;
  120. end
  121. end
  122. endmodule

 然后还需要一个发送模块,将数据发送给PC端,原理同接收模块大部分一致

  1. module uart_tx(
  2. //**************输入**************
  3. input clk, //系统50MHz时钟
  4. input rst_n, //系统复位,低电平有效
  5. input uart_en, //发送使能信号
  6. input [ 7:0] uart_din, //待发送数据
  7. //**************输出**************
  8. output uart_tx_busy, //发送忙状态标志
  9. output en_flag, //使能标志位
  10. output reg tx_flag, //发送过程标志信号
  11. output reg [ 7:0] tx_data, //寄存发送数据
  12. output reg [ 3:0] tx_cnt, //发送数据计数器
  13. output reg uart_txd //UART发送端口
  14. );
  15. //************参数定义************
  16. parameter CLK_FREQ = 50000000; //系统时钟频率
  17. parameter UART_BPS = 9600; //串口波特率
  18. localparam BPS_CNT = CLK_FREQ/UART_BPS; //波特率计数,串口传输一位所需要的系统时钟周期数
  19. //************信号定义************
  20. reg uart_en_1;
  21. reg uart_en_2;
  22. reg [15:0] clk_cnt; //系统时钟计数器
  23. //在串口发送过程中给出忙状态标志,其他模块就可以判断串口发送模块是否处于空闲状态
  24. assign uart_tx_busy = tx_flag;
  25. //捕获uart_en上升沿,得到一个时钟周期的脉冲信号,进入数据发送过程
  26. assign en_flag = (~uart_en_2) & uart_en_1;
  27. //对发送使能信号uart_en延迟两个时钟周期避免亚稳态
  28. always @(posedge clk or negedge rst_n) begin
  29. if (!rst_n) begin
  30. uart_en_1 <= 1'b0;
  31. uart_en_2 <= 1'b0;
  32. end
  33. else begin
  34. uart_en_1 <= uart_en;
  35. uart_en_2 <= uart_en_1;
  36. end
  37. end
  38. //当脉冲信号en_flag到达时,寄存待发送的数据,进入发送过程
  39. always @(posedge clk or negedge rst_n) begin
  40. if (!rst_n) begin
  41. tx_flag <= 1'b0;
  42. tx_data <= 8'd0;
  43. end
  44. //检测到发送使能上升沿,进入发送过程,标志位tx_flag拉高,寄存待发送数据
  45. else if (en_flag) begin
  46. tx_flag <= 1'b1;
  47. tx_data <= uart_din;
  48. end
  49. //计数到停止位结束时,停止发送过程,标志位tx_flag拉低
  50. else if ((tx_cnt == 4'd9) && (clk_cnt == BPS_CNT - (BPS_CNT/16))) begin //提前1/16个停止位拉低,保证发送数据时间略小于接收数据时间,避免数据积累造成丢失
  51. tx_flag <= 1'b0;
  52. tx_data <= 8'd0;
  53. end
  54. else begin
  55. tx_flag <= tx_flag;
  56. tx_data <= tx_data;
  57. end
  58. end
  59. //进入发送过程后,启动系统时钟计数器
  60. always @(posedge clk or negedge rst_n) begin
  61. if (!rst_n)
  62. clk_cnt <= 16'd0;
  63. else if (tx_flag) begin
  64. if (clk_cnt < BPS_CNT - 1)
  65. clk_cnt <= clk_cnt + 1'b1;
  66. else
  67. clk_cnt <= 16'd0;
  68. end
  69. else
  70. clk_cnt <= 16'd0;
  71. end
  72. //进入发送过程后,启动发送数据计数器
  73. always @(posedge clk or negedge rst_n) begin
  74. if (!rst_n)
  75. tx_cnt <= 4'd0;
  76. else if (tx_flag) begin
  77. if (clk_cnt == BPS_CNT - 1)
  78. tx_cnt <= tx_cnt + 1'b1;
  79. else
  80. tx_cnt <= tx_cnt;
  81. end
  82. else
  83. tx_cnt <= 4'd0;
  84. end
  85. //根据发送数据计数器来给uart发送端口赋值
  86. always @(posedge clk or negedge rst_n) begin
  87. if (!rst_n)
  88. uart_txd <= 1'b1;
  89. else if (tx_flag)
  90. case(tx_cnt)
  91. 4'd0: uart_txd <= 1'b0; //起始位
  92. 4'd1: uart_txd <= tx_data[0]; //数据位最低位
  93. 4'd2: uart_txd <= tx_data[1];
  94. 4'd3: uart_txd <= tx_data[2];
  95. 4'd4: uart_txd <= tx_data[3];
  96. 4'd5: uart_txd <= tx_data[4];
  97. 4'd6: uart_txd <= tx_data[5];
  98. 4'd7: uart_txd <= tx_data[6];
  99. 4'd8: uart_txd <= tx_data[7]; //数据位最高位
  100. 4'd9: uart_txd <= 1'b1; //停止位
  101. default: ;
  102. endcase
  103. else
  104. uart_txd <= 1'b1; //空闲时发送端口为高电平
  105. end
  106. endmodule

环回模块将串口接收模块接收到的数据发送给串口发送模块

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

顶层文件

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

 参考资料:

正点原子FPGA教程、小梅哥FPGA教程

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/466291
推荐阅读
相关标签
  

闽ICP备14008679号