一、UART简介

  UART(universal asynchronous receiver-transmitter)是一种采用异步串行通信方式的通用异步收发传输器。一般来说,UART总是和RS232成对出现,那RS232又是什么呢? RS232也就是我们计算机上的串口,它的全称是EIA-RS-232C (简称232,或者是RS232 )。其中EIA(Electronic Industry Association)代表美国电子工业协会,RS是Recommended Standard的缩写,代表推荐标准,232 是标识符,C表示修改次数,它被广泛用于计算机串行接口外设连接。如果你的计算机上还有串口的话,那么你就可以在主机箱后面看到RS232的接口:

  随着时代的发展,这种借口已经很少用了,取而代之的是“USB转串口”,功能和原先一样,但接口更高效了。

  串口的主要功能为:在发送数据时将并行数据转换成串行数据进行传输,在接收数据时将接收到的串行数据转换成并行数据。这应该是大多数人接触电子后学习到的第一个通信协议吧。

二、通信格式

  下面来说说串口的具体要点:

1.传输时序

  UART串口通信需要两个信号线来实现,一根用于串口发送,另外一根负责串口接收。一开始高电平,然后拉低表示开始位,接着8个数据位,然后校验位,最后拉高表示停止位,并且进入空闲状态,等待下一次的数据传输。

  很多时候我们的校验位是允许省略的,所以协议就变成了:开始+数据+停止。

2.传输速率:波特率

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

  FPGA开发串口时,设计波特率的方法:FPGA的时钟频率/波特率。例如我的FPGA开发板时钟频率为50Mhz,即50_000_000hz,我想使用的波特率为9600bps,因此我需要的计数为:50000000/9600≈5208。

三、串口回环设计

  现在用FPGA开发板做一个串口回环的实验,要求是PC端通过串口助手发送数据给FPGA,FPGA接收到数据后返回给PC端,并在串口助手处显示数值。即串口助手发什么就能收回什么。实验框图如下:

1.uart_rx

 //**************************************************************************
// *** 名称 : uart_rx.v
// *** 作者 : xianyu_FPGA
// *** 博客 : https://www.cnblogs.com/xianyufpga/
// *** 日期 : 2019-01-10
// *** 描述 : 串口接收模块,计数9.5下,其中停止位0.5下
// 因为串口助手发送本次停止位和下次开始位中间没有留空闲位
// 若计满10下,则才结束本次传输下次数据就来了,会来不及接收
//************************************************************************** module uart_rx
//========================< 参数 >==========================================
#(
parameter CLK = 50_000_000 , //系统时钟,50Mhz
parameter BPS = , //波特率
parameter BPS_CNT = CLK/BPS //波特率计数
)
//========================< 端口 >==========================================
(
input wire clk , //时钟,50Mhz
input wire rst_n , //复位,低电平有效
input wire din , //输入数据
output reg [:] dout , //输出数据
output reg dout_vld //输出数据的有效指示
);
//========================< 信号 >==========================================
reg rx0 ;
reg rx1 ;
reg rx2 ;
wire rx_en ;
reg flag ;
reg [:] cnt0 ;
wire add_cnt0 ;
wire end_cnt0 ;
reg [ :] cnt1 ;
wire add_cnt1 ;
wire end_cnt1 ;
reg [ :] data ; //==========================================================================
//== 消除亚稳态 + 下降沿检测
//==========================================================================
always @(posedge clk or negedge rst_n) begin
if(!rst_n) begin
rx0 <= ;
rx1 <= ;
rx2 <= ;
end
else begin
rx0 <= din;
rx1 <= rx0;
rx2 <= rx1;
end
end assign rx_en = rx2 && ~rx1; //==========================================================================
//== 接收状态指示
//==========================================================================
always @(posedge clk or negedge rst_n) begin
if(!rst_n)
flag <= ;
else if(rx_en)
flag <= ;
else if(end_cnt1)
flag <= ;
end //==========================================================================
//== 波特率计数
//==========================================================================
always @(posedge clk or negedge rst_n) begin
if(!rst_n)
cnt0 <= ;
else if(add_cnt0) begin
if(end_cnt0)
cnt0 <= ;
else
cnt0 <= cnt0 + ;
end
end assign add_cnt0 = flag;
assign end_cnt0 = cnt0== BPS_CNT- || end_cnt1; //==========================================================================
//== 开始1位(不接收) + 数据8位 + 停止0.5位(不接收),共10位
//==========================================================================
always @(posedge clk or negedge rst_n) begin
if(!rst_n)
cnt1 <= ;
else if(add_cnt1) begin
if(end_cnt1)
cnt1 <= ;
else
cnt1 <= cnt1 + ;
end
end assign add_cnt1 = end_cnt0;
assign end_cnt1 = cnt1==- && cnt0==BPS_CNT/-; //==========================================================================
//== 缓存数据
//==========================================================================
always @ (posedge clk or negedge rst_n)begin
if(!rst_n)
data <= 'd0;
else if(cnt1>= && cnt1<= && cnt0==BPS_CNT/-) //中间采样
data[cnt1-] <= rx2; //或 dout <= {rx2,dout[7:1]};
end //==========================================================================
//== 输出数据
//==========================================================================
always @ (posedge clk or negedge rst_n)begin
if(!rst_n)
dout <= ;
else if(end_cnt1)
dout <= data;
end always @ (posedge clk or negedge rst_n)begin
if(!rst_n)
dout_vld <= ;
else if(end_cnt1)
dout_vld <= ;
else
dout_vld <= ;
end endmodule

2.uart_tx

 //**************************************************************************
// *** 名称 : uart_tx.v
// *** 作者 : xianyu_FPGA
// *** 博客 : https://www.cnblogs.com/xianyufpga/
// *** 日期 : 2019-01-10
// *** 描述 : 串口接收模块,计数9.5下,其中停止位0.5下
// 因为极端情况是本次停止位和下次开始位中间没有留空闲位
// 若计满10下,则才结束本次传输下次数据就来了,会来不及发送
//************************************************************************** module uart_tx
//========================< 参数 >==========================================
#(
parameter CLK = 50_000_000 , //系统时钟,50Mhz
parameter BPS = , //波特率
parameter BPS_CNT = CLK/BPS //波特率计数
)
//========================< 端口 >==========================================
(
input wire clk , //时钟,50Mhz
input wire rst_n , //复位,低电平有效
input wire [:] din , //输入数据
input wire din_vld , //输入数据的有效指示
output reg dout //输出数据
);
//========================< 信号 >==========================================
reg flag ;
reg [ :] din_tmp ;
reg [:] cnt0 ;
wire add_cnt0 ;
wire end_cnt0 ;
reg [ :] cnt1 ;
wire add_cnt1 ;
wire end_cnt1 ;
wire [ :] data ; //==========================================================================
//== 数据暂存(din可能会消失,暂存住)
//==========================================================================
always @ (posedge clk or negedge rst_n) begin
if(!rst_n)
din_tmp <='d0;
else if(din_vld)
din_tmp <= din;
end //==========================================================================
//== 发送状态指示
//==========================================================================
always @(posedge clk or negedge rst_n)begin
if(!rst_n)
flag <= ;
else if(din_vld)
flag <= ;
else if(end_cnt1)
flag <= ;
end //==========================================================================
//== 波特率计数
//==========================================================================
always @(posedge clk or negedge rst_n) begin
if(!rst_n)
cnt0 <= ;
else if(add_cnt0) begin
if(end_cnt0)
cnt0 <= ;
else
cnt0 <= cnt0 + ;
end
end assign add_cnt0 = flag;
assign end_cnt0 = cnt0== BPS_CNT- || end_cnt1; //==========================================================================
//== 开始1位 + 数据8位 + 停止0.5位,共10位
//==========================================================================
always @(posedge clk or negedge rst_n) begin
if(!rst_n)
cnt1 <= ;
else if(add_cnt1) begin
if(end_cnt1)
cnt1 <= ;
else
cnt1 <= cnt1 + ;
end
end assign add_cnt1 = end_cnt0;
assign end_cnt1 = cnt1==- && cnt0==BPS_CNT/-; //==========================================================================
//== 数据输出(用case语句也行)
//==========================================================================
assign data = {'b1,din_tmp,1'b0}; //停止,数据,开始 always @(posedge clk or negedge rst_n) begin
if(!rst_n)
dout <= 'b1;
else if(flag)
dout <= data[cnt1];
end endmodule

3.top层

 //**************************************************************************
// *** 名称 : uart_top.v
// *** 作者 : xianyu_FPGA
// *** 博客 : https://www.cnblogs.com/xianyufpga/
// *** 日期 : 2019-01-10
// *** 描述 : 串口实验顶层文件
//************************************************************************** module uart_top
//========================< 端口 >==========================================
(
input wire clk , //时钟,50Mhz
input wire rst_n , //复位,低电平有效
input wire uart_rx , //FPGA通过串口接收的数据
output wire uart_tx //FPGA通过串口发送的数据
); //========================< 连线 >==========================================
wire [:] data ;
wire data_vld ; //==========================================================================
//== 模块例化
//==========================================================================
uart_rx
#(
.BPS_CNT ( ) //仿真用
)
u_uart_rx
(
.clk (clk ),
.rst_n (rst_n ),
.din (uart_rx ),
.dout (data ),
.dout_vld (data_vld )
); uart_tx
#(
.BPS_CNT ( ) //仿真用
)
u_uart_tx
(
.clk (clk ),
.rst_n (rst_n ),
.din_vld (data_vld ),
.din (data ),
.dout (uart_tx )
); endmodule

四、仿真调试

1、testbench

 `timescale 1ns/1ps  //时间精度
`define Clock //时钟周期 module uart_top_tb; //========================< 端口 >==========================================
reg clk ; //时钟,50Mhz
reg rst_n ; //复位,低电平有效
reg uart_rx ;
wire uart_tx ; //==========================================================================
//== 模块例化
//==========================================================================
uart_top u_uart_top
(
.clk (clk ),
.rst_n (rst_n ),
.uart_rx (uart_rx ),
.uart_tx (uart_tx )
); //==========================================================================
//== 时钟信号和复位信号
//==========================================================================
initial begin
clk = ;
forever
#(`Clock/) clk = ~clk;
end initial begin
rst_n = ; #(`Clock*+);
rst_n = ;
end //==========================================================================
//== task任务
//==========================================================================
reg [:] mem[:] ; //位宽为8,深度为16个数据
integer i ;
integer j ; //读取外部数据
initial $readmemh("./data.txt",mem); //位赋值
task rx_bit
(
input [:] data
);
begin
for(i=;i<=;i=i+) begin //10个bit为
case(i)
: uart_rx = 'b0;
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = data[i-];
: uart_rx = 'b1;
endcase
#; //一个完整波特延时:52*20=1040
end //考虑到空闲位,也可以设置得1040稍大一些
end
endtask //字节赋值
task rx_byte;
begin
for(j=;j<=;j=j+) //16个byte数据
rx_bit(mem[j]);
end
endtask //==========================================================================
//== 调用task
//==========================================================================
initial begin
#(`Clock*+);
rx_byte();
end initial begin
#;
$stop;
end endmodule

2、data.txt

  testbench中调用了一个 data.txt 文本文档,里面存储了此次仿真的16个数据,将其放置到 Modelsim 软件的工程目录中(非 work)即可。


a
b
c
d
e
f

3、仿真波形

  由波形可以看到,本次设计应该是成功的。

五、上板验证

  本次上位机采用友善串口助手,无校验位,停止位为1。当串口助手发送数据给FPGA后,FPGA很快又将原数据返回给上位机。

  经上板验证,本次设计成功!

参考资料:

[1]明德扬FPGA教程

[2]正点原子FPGA教程

[2]威三学院FPGA教程

协议——UART(RS232)的更多相关文章

  1. UART\RS232与RS485的关系

    https://blog.csdn.net/lhl161123/article/details/53510593 串口通讯是电子工程师面对的最基本的一个通讯方式,RS-232是其中最简单的一种.然而, ...

  2. UART RS232 的CTS与RTS

    目前较为常用的串口有9针串口(DB9)和25针串口(DB25),通信距离较近时(<12m),可以用电缆线直接连接标准RS232端口(RS422,RS485较远),若距离较远,需附加调制解调器(M ...

  3. z-stack协议uart分析(DMA)

    1.从ZMain里面的main函数开始分析 2.进入int main( void ); HalDriverInit();   //硬件相关初始化,有DMA初始化和UART初始化 3.进入HalDriv ...

  4. 串行通讯协议--起止式异步通讯协议(UART)

    起止式异步通讯协议: 特点与格式: 起止式异步协议的特点是一个字符一个字符传输,并且传送一个字符总是以起始位开始,以停止位结束,字符之间没有固定的时间间隔要求.其格式如图3 所示.每一个字符的前面都有 ...

  5. v3学院带你一次性认清UART、RS-232、RS-422、RS-485的区别

    通讯问题,和交通问题一样,也有高速.低速.拥堵.中断等等各种情况.如果把串口通讯比做交通,UART比作车站,那么一帧的数据就好比汽车.汽车跑在路上,要遵守交通规则.如果是市内,一般限速30.40,而高 ...

  6. [转]UART通信简介

    1.前言 UART通信,即通用异步收发传输器(Universal Asynchronous Receiver/Transmitter). 串行通信是指利用一条传输线将资料一位位地顺序传送.特点是通信线 ...

  7. RTC实时时间系统学习笔记(一)---------------UART串口

    临近研三了,自己倾向于要找数字IC方面的工作,苦于教研室的项目一直都是调板子调板子调板子,真正用到FPGA的很少,,本着"工欲善其事必先利其器"的原则,在网上搜寻如何自学FPGA, ...

  8. USB、UART、SPI等总线速率

    1. USB总线 USB1.1: ---低速模式(low speed):1.5Mbps ---全速模式(full speed): 12Mbps USB2.0:向下兼容.增加了高速模式,最大速率480M ...

  9. USB、UART、SPI等总线速率(转)

    1. USB总线 USB1.1: ——-低速模式(low speed):1.5Mbps ——-全速模式(full speed): 12Mbps USB2.0:向下兼容.增加了高速模式,最大速率480M ...

随机推荐

  1. 使用vault pki engine 方便的管理证书

    vault 是一个很方便的secret .敏感数据管理工具,当前的版本已经包含了UI,使用起来很方便 以下演示一个简单的pki 管理 项目使用docker-compose 运行,为了简单使用单机开发模 ...

  2. 什么是uni-app?

    uni-app 是一个使用 Vue.js 开发所有前端应用的框架,开发者编写一套代码,可发布到iOS.Android.H5.以及各种小程序(微信/支付宝/百度/头条/QQ/钉钉)等多个平台. 即使不跨 ...

  3. 关于 Mercury_Lc 说明

    现在还主要在用 csdn 写博客,博客地址:https://blog.csdn.net/Mercury_Lc 这个是因为好奇,点了一下 一键搬家 ,就酱紫了. 主要更新,前往这个网址 https:// ...

  4. 【2019.11.27】SDN课程阅读作业(2)

    过去20年中可编程网络的发展可以分为几个阶段?每个阶段的贡献是什么? Making computer networks more programmable enables innovation in ...

  5. Spring Boot 配置文件 bootstrap vs application 到底有什么区别?

    用过 Spring Boot 的都知道在 Spring Boot 中有以下两种配置文件 bootstrap (.yml 或者 .properties) application (.yml 或者 .pr ...

  6. Apache Flink - 架构和拓扑

    Flink结构: flink cli 解析本地环境配置,启动 ApplicationMaster 在 ApplicationMaster 中启动 JobManager 在 ApplicationMas ...

  7. 【转】使用fastboot命令刷机流程详解

    一.Fastboot是什么? 1.1 首先介绍Recovery模式(卡刷) 在系统进行定制时,编译系统会编译出一份ZIP的压缩包,里面是一些系统分区镜像,提供给客户进行手动升级.恢复系统.需要提前将压 ...

  8. mysql-connector-java(6.0以上)的时差问题

    一.背景 通过mybatis日志观察插入数据库的时间为当前时间,但是打开数据库表发现时间滞后了8个小时. 二.推论及解决 很容易猜到这是时区的问题. 三.最后找到的问题点如下: jdbc:mysql: ...

  9. vue使用formData进行文件上传

    本文为博主原创,未经允许不得转载 1.vue页面 <ux-form ref="formRef" layout="vertical"> <ux- ...

  10. [转]JS - Promise使用详解2(ES6中的Promise)

    原文地址:https://www.hangge.com/blog/cache/detail_1638.html 2015年6月, ES2015(即 ECMAScript 6.ES6) 正式发布.其中  ...