【FPGA入门】第六篇、异步串口通信

news/2024/11/8 2:51:08/

目录

第一部分、相关知识     

1、UART和RS232的区别

2、UART与USART的区别

3、全双工?

4、RS232通信协议

5、波特率

6、如何将外部异步信号变为内部同步信号?

7、什么时间点让FPGA去采集rx线上的数据?

第二部分、串口通信时序图

1、发送线RX流程

1.1、模块图

1.2、端口介绍

1.3、时序波形图

1.4、RX接收代码

2、接收线TX流程

2.1、模块图

2.2、时序波形图

2.3、 TX发送代码

3、Top层代码

第三部分、仿真代码的编写

1、testbench代码

 2、仿真结果

3、上板测试

 第四部分、总结


第一部分、相关知识     

1、UART和RS232的区别

  • UART就是一堆电路,是异步串行通信的一种电路实现;
  • RS232属于异步串行通信方式,跟UART相比,区别仅在于UART使用的是TTL电平,而RS232使用的是另一种电平标准。

2、UART与USART的区别

UART:UART 的全称叫做通用异步收发传输器。

  • 将数据在串行通信和并行通信间的传输转换。通俗的讲就是把多比特的数据转化为单比特的数据,或者把单比特的数据转化为多比特的数据。工作原理是将数据的每个 bit 一位接一位地传输。
  • UART 是一种通用串行数据总线,用于异步通信。该总线双向通信,可以实现全双工传输和接收

USART:通用同步和异步收发器

  • 当进行异步通信时,这两者是没有区别的。
  • 区别在于USART比UART多了同步通信功能,因此USART可以提供一个主动的时钟线。使用外部时钟使 USART 的数据速率远高于标准 UART 的数据速率。

3、全双工?

        全双工:表示发送和接收不会相互影响。

4、RS232通信协议

        rs232 是 uart 的一种,有两根线,分别是 rx 和 tx,这两根线都是 1 比特位宽的。其中 rx 是接收线, tx 是发送线。

  • rx,位宽为 1 比特, pc 机通过串口往 FPGA 发 8 比特数据时, FPGA 通过串口线 rx 一位一位地接收,从最低位到最高位依次接收,最后在 FPGA 里面位拼接成8 比特数据。
  • tx,位宽为 1 比特, FPGA 通过串口往 pc 机发 8 比特数据时, FPGA 把 8 比特数据通过 tx 线一位一位的传给 pc 机,从最低位到最高位依次发送,最后上位机通过串口助手把这一位一位的数据位拼接成 8 比特数据。

        在不发送或者不接收数据的情况下, rx 和 tx 处于空闲状态,此时 rx 和 tx 线都保持高电平

        如果有数据传递,首先会有一个起始位(0),然后是 8 比特的数据位,接着有 1 比特的停止位

        如果停止位以后不再发数据,将进入空闲状态,否则又将数据线拉低(进入起始位状态)。

        注意:起始位和停止位各算一位,因此一共是10位数据,而停止位不需要去做判断。

5、波特率

        波特率在串口通信时的速率,单位时间内载波变化的次数,这里选用的是9600Bd,即发送一比特数据需要的时间为 1/9600 秒。

        用串口发送或者接收数据(起始位、数据位、停止位)时,每发送或者接收一位数据的时间都需要 1 个波特,即 1/9600 秒。

        串口发送或者接收一比特数据的时间为一个波特(1/9600),因此如果用 50M的系统时钟来计数,就需要记数 cnt=(1/9600s)/20ns≈5208 个系统时钟,才再次发送或者接收下一个数据。

        注意:cnt=(1/9600s)/20ns≈5208.33 ,累计误差0.33 * 10 = 3ns,这个误差太小了,是不会影响到传输的。而且计数器每次都是从0开始计数,因此,该误差也不会累加。

        上位机通过串口发 8 比特数据时,会自动在发 8 位有效数据前发一个波特时间的起始位,也会自动在发完 8 位有效数据后发一个停止位。同理,串口助手接收 fpga 发送的数据前,必须检测到一波特的起始位才会接收数据,接收完数据后,再接收一个停止位,所以 FPGA 通过串口除了发数据以外,还要发起始位和停止位。

6、如何将外部异步信号变为内部同步信号?

        通过前面的介绍可知,RS232为通用异步收发传输器,那么rx传到FPGA内部的数据与系统的时钟是不匹配的,没有任何逻辑关系,因此要借助寄存器将rx进行打两拍操作后,才能用来进行逻辑运算。

        这么做的目的是尽量降低亚稳态,如果不做这样 的处理,就可能导致采集的数据不稳定,只有做了亚稳态处理后的数据才在后面使用。

7、什么时间点让FPGA去采集rx线上的数据?

        外部的数据通过 rx 线传到 FPGA, FPGA 采数据时,在中间时刻采数可以保证采的数据最 稳定。

第二部分、串口通信时序图

1、发送线RX流程

        1.1、模块图

         1.2、端口介绍

         1.3、时序波形图

        1.4、RX接收代码

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : uart_rx.v
// Create : 2023-06-20 12:45:49
// -----------------------------------------------------------------------------
module uart_rx(input wire sclk,input wire s_rst_n,input wire rx,output reg [7:0]po_data,output reg po_flag);
// //9600波特率参量
// parameter BAUD_END = 5207;//波特率9600,接收间隔最大计数MAXEND = 50M/9600 = 5208.33 - 1 = 5207
// parameter BAUD_HALF = 2602;//5208/2 - 1 = 2603,要在2603这里点看到波形
// parameter BAUD_BIT_WIDTH = 12;//5207需要13位来保存,[12:0]//115200波特率参量
parameter BAUD_END = 433;//波特率115200,接收间隔最大计数MAXEND = 50M/115200 = 434.02 - 1 = 433
parameter BAUD_HALF = 216;//434/2 - 1 = 216
parameter BAUD_BIT_WIDTH = 8;//5207需要9位来保存,[8:0]//变量
reg rx1;//用来降低亚稳态
reg rx2;//用来降低亚稳态
reg rx2_reg;//用来打拍rx2,判断下降沿reg rx_flag;//rx信号有效标志位reg [12:0]baud_cnt;//9600波特率的1个波特时间1/9600s的计数器reg capture_flag;//中间采集数据的时间点reg [3:0]capture_cnt;//采集数据的计数器//利用移位寄存器打拍,打拍信号可以不加复位
always @(posedge sclk) begin{rx2_reg,rx2,rx1} <= {rx2,rx1,rx};
end//rx有效标志位,rx_flag
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginrx_flag <= 1'b0;end//rx_flag受baud_cnt影响,究竟是什么样的影响?else if(capture_flag == 1'b1 && capture_cnt == 'd8)beginrx_flag <= 1'b0;endelse if (rx2 == 1'b0 && rx2_reg == 1'b1) beginrx_flag <= 1'b1;end
end//一波特时间对应的计数器
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginbaud_cnt <= 'd0;endelse if(baud_cnt == BAUD_END)beginbaud_cnt <= 'd0;end//baud_cnt会计数到2603,若判断条件换成rx_flag == 1'b0,baud_cnt会计数到2604else if(capture_flag == 1'b1 && capture_cnt == 'd8)beginbaud_cnt <= 'd0;endelse if (rx_flag == 1'b1) beginbaud_cnt <= baud_cnt + 1'b1;end
end//中间采集数据的时间点标志
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) begincapture_flag <= 1'b0;endelse if (baud_cnt == BAUD_HALF) begincapture_flag <= 1'b1;endelse begincapture_flag <= 1'b0;end
end//采集数据的计数
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) begincapture_cnt <= 'd0;endelse if (capture_cnt == 'd8 && capture_flag == 1'b1) begincapture_cnt <= 'd0;endelse if(capture_flag == 1'b1)begincapture_cnt <= capture_cnt + 1'b1;end
end//输出po_data
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginpo_data <= 'd0;endelse if (capture_cnt >= 1'b1 && capture_flag == 1'b1) beginpo_data <= {rx2_reg,po_data[7:1]};end
end//输出结束标志
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginpo_flag <= 1'b0;endelse if (capture_cnt == 'd8 && capture_flag == 1'b1) beginpo_flag <= 1'b1;endelse beginpo_flag <= 1'b0;end
endendmodule

2、接收线TX流程

        2.1、模块图

         2.2、时序波形图

        2.3、 TX发送代码

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : uart_tx.v
// Create : 2023-06-20 14:14:25
// -----------------------------------------------------------------------------
module uart_tx(input wire sclk,input wire s_rst_n,input wire [7:0]pi_data,input wire pi_flag,output reg tx);
// //9600波特率参量
// parameter BAUD_END = 5207;//波特率9600,接收间隔最大计数MAXEND = 50M/9600 = 5208.33 - 1 = 5207//115200波特率参量
parameter BAUD_END = 433;//波特率115200,接收间隔最大计数MAXEND = 50M/115200 = 434.02 - 1 = 433//变量
reg [7:0]pi_data_reg;//保存输入的数据
reg tx_flag;		 //发送数据的有效位,服务于计数器 
reg [12:0]baud_cnt;  //9600波特率的1个波特时间1/9600s的计数器
reg bit_flag;		 //计满flag
reg [3:0]bit_cnt;	 //发送数据计数。//保存输入的数据,因为po_data在接收数据的时候一直在变
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginpi_data_reg <= 'd0;endelse if(pi_flag == 1'b1)begin//这里是寄存的作用,不是直接传过来pi_data_reg <= pi_data;end
end//发送数据有效时间段
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) begintx_flag <= 1'b0;endelse if(bit_flag == 1'b1 && bit_cnt == 'd8)begintx_flag <= 1'b0;endelse if (pi_flag == 1'b1) begintx_flag <= 1'b1;end
end//一波特时间对应的计数器
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginbaud_cnt <= 'd0;endelse if (baud_cnt == BAUD_END) beginbaud_cnt <= 'd0;endelse if(tx_flag == 1'b1)beginbaud_cnt <= baud_cnt + 1'b1; end
end//bit_flag
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginbit_flag <= 1'b0;endelse if (baud_cnt == BAUD_END - 1'b1) beginbit_flag <= 1'b1;endelse beginbit_flag <= 1'b0;end
end//发送数据计数
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) beginbit_cnt <= 'd0;endelse if(bit_cnt == 'd8 && bit_flag == 1'b1)beginbit_cnt <= 'd0;endelse if (bit_flag == 1'b1) beginbit_cnt <= bit_cnt + 1'b1;end
end//发送数据
always @(posedge sclk or negedge s_rst_n) beginif (!s_rst_n) begintx <= 1'b1;endelse if(pi_flag == 1'b1)begintx <= 1'b0;//起始位,判断到有输入,立马拉低endelse if (bit_flag == 1'b1 && bit_cnt <= 'd7) begintx <= pi_data_reg[bit_cnt];endelse if(bit_flag == 1'b1 && bit_cnt == 'd8)begintx <= 1'b1;//停止位end
endendmodule

3、Top层代码

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : top.v
// Create : 2023-06-20 14:31:06
// -----------------------------------------------------------------------------
module top(input wire clk,input wire rst_n,input wire rx,output wire tx);wire [7:0]data;
wire flag;uart_rx inst_uart_rx (.sclk    (clk),.s_rst_n (rst_n),.rx      (rx),.po_data (data),.po_flag (flag));uart_tx  inst_uart_tx (.sclk    (clk),.s_rst_n (rst_n),.pi_data (data),.pi_flag (flag),.tx      (tx));endmodule

第三部分、仿真代码的编写

        注意:在testbench同文件位置建立如下文本

00000001
00000010
00000011
00000100
00000101
00000110
00000111
00001000
00001001
00001010
00001011
00001100
00001101
00001110
00001111
11111111

1、testbench代码

          关于top层的testbench代码如下: 

// -----------------------------------------------------------------------------
// Copyright (c) 2014-2023 All rights reserved
// -----------------------------------------------------------------------------
// Author : BigFartPeach
// CSDN   : 大屁桃
// E-mail : 2624507313@qq.com
// File   : top_sim.v
// Create : 2023-04-25 11:24:00
// -----------------------------------------------------------------------------
module top_sim();reg clk;
reg rst_n;
reg rx;
wire tx;reg [7:0]mem[15:0];initial beginclk = 0;rst_n = 0;rx = 1;#100;rst_n = 1;
endalways #10 clk = ~clk;initial begin$readmemb("./data.txt",mem);
endinitial begin#200;read_text();
end//多次读文本数据
task read_text();integer i;beginfor(i = 0; i < 16; i = i + 1)beginsend_bit_data(mem[i]);endend
endtasktask send_bit_data(input [7:0]data_txt);integer i;beginfor(i = 0; i < 10;i = i + 1)begincase (i)0: rx = 0;//起始位1: rx = data_txt[0];//rx = mem[i - 1];	2: rx = data_txt[1];//rx = mem[i - 1];	3: rx = data_txt[2];//rx = mem[i - 1];	4: rx = data_txt[3];//rx = mem[i - 1];	5: rx = data_txt[4];//rx = mem[i - 1];	6: rx = data_txt[5];//rx = mem[i - 1];	7: rx = data_txt[6];//rx = mem[i - 1];	8: rx = data_txt[7];//rx = mem[i - 1];	9: rx = 1;//停止位endcase#104166;//1/9600波特时间endend
endtasktop inst_top (.clk(clk), .rst_n(rst_n),.rx(rx),.tx(tx));endmodule

 2、仿真结果

        可以看到rx与tx的波形一致。

3、上板测试

        波特率为9600发送后返回的数据

             波特率为115200发送后返回的数据

 第四部分、总结

        本篇博客介绍了串口的基本知识以及驱动代码的编写,博客中若有错误欢迎大家及时私信我。

        最后,希望我的博客对你有帮助😎😎😎😎,有需要的小伙伴可以查看本专栏更多的往期文章专栏链接如下:FPGA的学习之旅_大屁桃的博客-CSDN博客

        博客中涉及到的工程链接如下,工程基于ISE软件,没有积分的小伙伴评论留下邮箱即可

        FPGA入门第六篇、异步串口通信工程资源-CSDN文库


http://www.ppmy.cn/news/485305.html

相关文章

node 跟踪异步资源(回调)async_hook、AsyncResource

文章目录 跟踪异步回调的原因异步调用链基础async_hookAsyncResourceasyncResource.runInAsyncScope 跟踪异步回调的原因 node 基于事件循环的异步非阻塞 I/O 模型&#xff0c;发起一次异步调用&#xff0c;回调在之后的循环中才被调用&#xff0c;此时已经无法追踪到是谁发起…

vue中 @scroll的使用

@scroll 是 Vue 中的一个修饰符,可以用来监听元素的滚动事件。 使用方法: 在元素上绑定一个 @scroll 修饰符,并绑定一个事件处理函数,如:<template><div class="container" @scroll="handleScroll"></div> </template><sc…

Vue点击上下滑动滚屏

上滑到顶部 绑定点击事件&#xff1a; <div class"totop"><a click"backTop()" size"50" v-show"tabshow" class"btn btn-default"><span class"fa"></span></a></div> 点…

无缝滚动:vue-seamless-scroll

1、安装 npm install vue-seamless-scroll --save2、注册 // 方法一&#xff1a;main.js中全局注册 import vueSeamlessScroll from vue-seamless-scroll Vue.use(vueSeamlessScroll)// 方法二&#xff1a;局部注册 import vueSeamlessScroll from vue-seamless-scroll expor…

vue:使用vuescroll

1、npm下载 npm install vuescroll -S 官网&#xff1a;https://vuescrolljs.yvescoding.org/zh/ <template><div><vuescroll :ops"ops"><div class"container"><ul v-for"(item, index) in 100" :key"index&q…

vue 简单的tab切换滑动效果

实现一个简单的滑动效果&#xff1a; 代码逻辑比较简单&#xff0c;利用css的transform和transition属性实现简单的tab切换效果 贴上代码&#xff0c;仅供参考&#xff1a; html部分&#xff1a; <template><div class"container"><div class"…

【Vue实用功能】Vue中实现移动端的scroll滚动

Vue中实现移动端的scroll滚动 介绍&#xff1a; 在移动端或PC&#xff0c;页面的部分内容需要我们让其在页面滚动&#xff0c;这时候我们都会使用::-webkit-scrollbar来修饰原生滚动条&#xff0c;这样会影响滚动条对宽度的检测&#xff0c;所有就有了该组件&#xff0c;不需要…

Vue【vue-seamless-scroll】滚动组件及注意事项

一、运用场景&#xff1a;VUE开发的项目中需要表格滚动展示信息 二、组件&#xff1a;vue-seamless-scroll 三、使用&#xff1a;通过ul标签css样式模拟表格&#xff0c;表头表数据 安装 npm install vue-seamless-scroll --save在main.js中引入 import scroll from vue-s…