首页 > 代码库 > uart通讯协议

uart通讯协议

本次设计的源码在http://download.csdn.net/detail/noticeable/9912383 下载

实验目的:通过uart通讯协议的编写,了解FPGA的通讯协议编写的方法。

 

实验现象:FPAG可以通过USB转TTL工具向电脑收发数据。

 

相关知识点:1、uart通讯协议是什么及其内容。2、in system surce and probes(editor)调试工具的使用。

  

 

关于串行通讯:串口通讯协议是一种主机之间常用的通讯协议,通过模块按位发送和接收字节,可以达到通讯的目的,其通讯只需要三根根数据线(GND,RXD,TXD)即可完成。其中串口通讯最重要的参数是波特率,数据位,停止位和奇偶校验位,下面来一一讲解这                              些参数的作用。

        (1)波特率:波特率是串口的通讯速率,常见的比特率为1200bps、4800bps、9600bps、38400bps、115200bps、256000bps、500000bps,这里bps的意思是bit/s,因此可以知道,波特率实际上是每秒可以传输的bit的个数,由此可知波特率与时钟是直接挂钩的,比如波特率位9600bps,那么时钟就是9600hz,即每秒内产生9600个时钟,每个时钟周期发送1bit的数据。(这里注意:波特率越高,传输距离越短)

          波特率分频计数器的方法:技术分享

 

 

        (2)数据位:数据位可以在通讯过程中告知另一个主机,完成一次通讯过程中真正有效的数据是多少位,完成对传输数据的位数的校验。

        (3)起始位、停止位:起始位位于单个数据包的第一位,停止位位于单个包的最后一位,每个包都是一个字节, 另一台主机接收到停止位时就知道信号发送已经完成了。

         (4)奇偶校验位:奇偶校验是通讯协议中的一种简单的检错方法,其有时钟检错方式,:偶、奇、高和低。当然没有校验位也是可以的。对于偶和奇校验的情况,串口会设置校验位(数据位后面的一位),用一个值确保传输的数据有偶个或者奇个逻辑高位。例如,如果数据是011,那么对于偶校验,校验位为0,保证逻辑高的位数是偶数个。如果是奇校验,校验位为1,这样就有3个逻辑高位。高位和低位不真正的检查数据,简单置位逻辑高或者逻辑低校验。这样使得接收设备能够知道一个位的状态,有机会判断是否有噪声干扰了通信或者是否传输和接收数据是否不同步。

 

 

项目设计:

本次设计发送模块的整体结构体如下                                       发送时序图

技术分享技术分享

 

 

  首先按照系统框图编写tx项目源码:

 

      技术分享

发生时序图

 

 源码程序如下

 

module uart_tx(
                                clk,
                                rst_n,
                               send_en,
                                baud_set,
                                tx_done,
                                rs232_tx,
                                data_byte,
                                uart_state
                            );
        input clk;
        input rst_n;
        input [7:0]data_byte;//数据发送寄存器
        input send_en;
        input [2:0]baud_set;
        
        output reg rs232_tx;                    //输出引脚
        output reg tx_done;                    //传输完成的标志位
        output reg uart_state;                //uart_state传送状态
        
        reg [15:0]div_cnt;        //分频计数器
        reg bps_clk;                    //波特率时钟
        reg [15:0]bps_dr;            //分频计数最大值
        reg [3:0]bps_cnt;            //波特率计数时钟
        reg [7:0]r_data_byte;//发送缓存区
        
        localparam start_bit=1b0;
      localparam stop_bit=1b1;
        
        //具有优先级的信号控制,产生控制信号
        always@(posedge clk or negedge rst_n)
                    if(!rst_n)
            uart_state<=0;
            else if(send_en==1b1)
            uart_state<=1;
            else if(bps_cnt == 4d11)                //假设穿的是一组8位的数据+起始位+停止位
        uart_state <= 1b0;
            else 
            uart_state<=uart_state;
            
                    //发送过程中需要保证数据是稳定的,选用一个寄存器将数据先缓存起来
            always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                    r_data_byte<=8d0;
                    else if(send_en)
                    r_data_byte<=data_byte;
        
                        //设计查找表DR_LUT,通过查找表设置波特率
              //1/波特率/20ns
always@(posedge clk or negedge rst_n) if(!rst_n) bps_dr<=16d5207; else begin case (baud_set) 0:bps_dr<=16d5207;//9600 1:bps_dr<=16d2603;//19200 2:bps_dr<=16d1301;//28400 3:bps_dr<=16d867;//57600 4:bps_dr<=16d433;//115200 // 5:bps_dr<=16‘d5207; // 6:bps_dr<=16‘d5207; // 7:bps_dr<=16‘d5207; default bps_dr<=16d5207;//9600 endcase end //分频计数器 always@(posedge clk or negedge rst_n) if(!rst_n) div_cnt<=16d0; else if(uart_state)begin if(div_cnt==bps_dr)//到达计数最大值时清零 div_cnt<=16d0; else div_cnt<=div_cnt+1b1; end else div_cnt<=16d0; //单周期的波特率时钟产生 always@(posedge clk or negedge rst_n) if(!rst_n) bps_clk<=1b0; else if(div_cnt==16d1) bps_clk<=1; else bps_clk<=0; //设计波特率计数器 always@(posedge clk or negedge rst_n) if(!rst_n) bps_cnt<=4b0; else if (bps_cnt == 4d11) bps_cnt<=4b0; else if (bps_clk) bps_cnt<=bps_cnt+1b1; else bps_cnt<=bps_cnt; //发送完成信号 always@(posedge clk or negedge rst_n) if(!rst_n) tx_done<=1b0; else if(bps_cnt==4d11) tx_done<=1b1; else tx_done <=1b0; //数据发送,即一个十选一的多路器 always@(posedge clk or negedge rst_n) if(!rst_n) rs232_tx<=1b1; else begin case(bps_cnt) 0:rs232_tx<=1b1; 1:rs232_tx<=start_bit;//起始位 2:rs232_tx<=r_data_byte[0]; 3:rs232_tx<=r_data_byte[1]; 4:rs232_tx<=r_data_byte[2]; 5:rs232_tx<=r_data_byte[3]; 6:rs232_tx<=r_data_byte[4]; 7:rs232_tx<=r_data_byte[5]; 8:rs232_tx<=r_data_byte[6]; 9:rs232_tx<=r_data_byte[7]; 10:rs232_tx<=stop_bit;//结束位,本次设计不设奇偶校验位 default rs232_tx<=1b1; endcase end endmodule

 

 

 编写testbench文件

`timescale 1ns/1ns
`define clock_period 20

module uart_tx_tb;
        reg clk;
        reg rst_n;
        reg [7:0]data_byte;
        reg send_en;
        reg [2:0]baud_set;
        
        wire rs232_tx;
        wire tx_done;
        wire uart_state;
        uart_tx            uart_tx0(
                                                            .clk(clk),
                                                            .rst_n(rst_n),
                                                            .send_en(send_en),
                                                           .baud_set(baud_set),
                                                            .tx_done(tx_done),
                                                            .rs232_tx(rs232_tx),
                                                            .data_byte(data_byte),
                                                            .uart_state(uart_state)
                                                        );
                                                        
                                initial clk=1;
                                always#(`clock_period/2)  clk=~clk;
                                initial begin 
                                rst_n<=1b0;
                                data_byte<=8d0;
                                send_en<=1d0;
                                baud_set=3d4;
                                #(`clock_period*20+1)
                                rst_n<=1b1;
                                    #(`clock_period*50+1)
                                    data_byte<=8haa;
                                send_en<=1d1;
                                #(`clock_period)
                                send_en<=1d0;
                                @(posedge tx_done)//等待传输完成的上升沿
                                #(`clock_period*500)//重新发送
                                data_byte<=8h55;
                                send_en<=1d1;
                                #(`clock_period)
                                send_en<=1d0;
                                #(`clock_period*500)
                                $stop;
                                
                                end
                                
                                
                                
                                
                            
        endmodule 

 

仿真结果如下,可以看到,每当有一个send_en 时,databyte都会将串口数据输出出来。

        技术分享

 

将之前用到的key_filter文件添加进来,作为传输的控制信号。

技术分享

 添加IP核以便使用in system sources and probe editor 工具

技术分享技术分享

 

 将.v文件添加到file中,编写uart_top

module uart_top(clk ,rst_n,rs232_tx,key_in,led);

                    input key_in;
                    input clk;
                    input rst_n;
                    output rs232_tx;
                    wire send_en;
                    wire [7:0]data_byte;
                    output led;
                    wire key_flag,key_state;
                    assign send_en=key_flag&!key_state;//按键检测成功且为低电平时,发生使能
                    
                uart_tx            uart_tx1(
                                                            .clk(clk),
                                                            .rst_n(rst_n),
                                                            .send_en(send_en),
                                                           .baud_set(3d0),
                                                            .tx_done(),
                                                            .rs232_tx(rs232_tx),
                                                            .data_byte(data_byte),
                                                            .uart_state(led)//将串口状态用LED显示出来
                                                        );
                                key_filter key_filter0(
                                                                .clk(clk),
                                                                .rst_n(rst_n),
                                                                .key_in(key_in),
                                                                .key_flag(key_flag),
                                                                .key_state(key_state)
                                                                );
                         ISSP  ISSP0(
                                                        .probe(),
                                                        .source(data_byte)
                                                        );
endmodule

 

 

 进行引脚分配,进行板级验证。

技术分享

由于DE1-SOC上并未板载USB转TTL模块,这里需要自备一个,然后将模块的RXD口连接到开发板上的GPIO_0D1口上,打开串口调试软件并通过USB转TTL模块连接到FPGA上(波特率设为9600),按下KEY1后,可以看到开发板上的LEDR0快速闪烁一下,串口调试助手接收到00两个数据

 

 技术分享

 

打开in system sources and probes editor,

技术分享

修改格式为hex格式,然后输入任意数据,我这里输入66,

技术分享

 然后再按一下key1,可以看到串口调试软件显示出来66,即我们给的source值,这里发送端的设计就完成了,下面继续按相同的方法设计接收端的协议。

技术分享

 

 技术分享

 

 

 

UART数据接收部分:

 之前已经讲过了uart 发送端的时序图,对应的,理论上接收端的时序图如下,一般采样在每一位数据的中点是最稳定的。

 技术分享

但是在工业等复杂的环境中,电磁场的干扰是很强的,所以在这里需要进行抗干扰处理,需要多次采样求概率来进行接收,进行改进后的单bit数据接收方式示意图

 

技术分享

 

同样编写Verilog代码:

module uart_rx(clk,
                                    rs232_rx,
                                    baud_set,
                                    rst_n,
                                    data_byte,
                                    rx_done
                                    );
                                    input clk;
                        input                rs232_rx;
                        input             [2:0]baud_set;
                        input             rst_n;
                        output    reg     [7:0]data_byte;
                        output     reg        rx_done;
                        reg s0_rs232_rx,s1_rs232_rx;//两个同步寄存器
                        reg tmp0_rs232_rx,tmp1_rs232_rx;//数据寄存器
                        wire nedege;
                        reg [15:0]bps_dr;//分频计数器计数最大值
                        reg [15:0]div_cnt;//分频计数器
                        reg uart_state;
                        reg bps_clk;
                        reg [7:0]bps_cnt;
                        reg [2:0]r_data_byte [7:0];//前面[2:0]是每一位数据的存储宽度,[7:0]指位宽
                            reg [2:0] start_bit,stop_bit;
                        
                        
                        //异步信号同步处理,消除亚稳态,有疑惑的看之前的按键消抖部分
                        always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                    begin
                    s0_rs232_rx<=1b0;
                        s1_rs232_rx<=1b0;
                        end
                        else 
                        begin 
                            s0_rs232_rx<=rs232_rx;
                        s1_rs232_rx<=s0_rs232_rx;
                        end
                                //数据寄存
                        always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                    begin
                    tmp0_rs232_rx<=1b0;
                        tmp1_rs232_rx<=1b0;
                        end
                        else 
                        begin 
                            tmp0_rs232_rx<=s1_rs232_rx;
                                    tmp1_rs232_rx<=tmp0_rs232_rx;
                        end
                        assign nedege=tmp0_rs232_rx&tmp1_rs232_rx;//下降沿检测
                        
                        
                        
                    //波特率设置模块
                    //10^9/波特率/20ns/这里为了稳定1bit数据会采16次所以还要除16
                    
                    always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                bps_dr<=16d324;
                    else begin 
                    case (baud_set)
                    0:bps_dr<=16d324;//9600
                    1:bps_dr<=16d162;//19200
                    2:bps_dr<=16d80;//28400
                    3:bps_dr<=16d53;//57600
                    4:bps_dr<=16d26;//115200
                        default
                        bps_dr<=16d324;//9600
                    endcase 
                    end
                    //分频计数器
        always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                    div_cnt<=16d0;
                    else if(uart_state)begin 
                    if(div_cnt==bps_dr)//到达计数最大值时清零
                            div_cnt<=16d0;
                            else div_cnt<=div_cnt+1b1;
                                                end
                        else 
                        div_cnt<=16d0;
    
                                //单周期的波特率时钟产生
                    always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                    bps_clk<=1b0;
                    else if(div_cnt==16d1)
                        bps_clk<=1;
                        else
                        bps_clk<=0;
            
                                    //设计波特率计数器
                always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                bps_cnt<=8b0;
            else if(bps_cnt == 8d159 | (bps_cnt == 8d12 && (start_bit > 2)))//到12位的时候,start_bit>2说明接收错误,起始位不对
                            bps_cnt <= 8d0;//接收完成或者开始检测到错误信号,波特率时钟停止
                else if (bps_clk)
                        bps_cnt<=bps_cnt+1b1;
                else 
                        bps_cnt<=bps_cnt;
            
                //接收完成信号
                always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                    rx_done<=1b0;
                    else if(bps_cnt==8d159)
                    rx_done<=1b1;
                    else 
                    rx_done <=1b0;
                    
                
                            //数据读取,对每次采样进行求和值 
                    always@(posedge clk or negedge rst_n)
                    if(!rst_n)begin 
                            start_bit=3d0;
                            r_data_byte[0]<=3d0;
                            r_data_byte[1]<=3d0;
                            r_data_byte[2]<=3d0;
                            r_data_byte[3]<=3d0;
                            r_data_byte[4]<=3d0;
                            r_data_byte[5]<=3d0;
                            r_data_byte[6]<=3d0;
                            r_data_byte[7]<=3d0;
                            stop_bit<=3d0;
                            end 
                            else if(bps_clk)begin
        case(bps_cnt)
            0:begin
                    start_bit = 3d0;
                    r_data_byte[0] <= 3d0;
                    r_data_byte[1] <= 3d0;
                    r_data_byte[2] <= 3d0;
                    r_data_byte[3] <= 3d0;
                    r_data_byte[4] <= 3d0;
                    r_data_byte[5] <= 3d0;
                    r_data_byte[6] <= 3d0;
                    r_data_byte[7] <= 3d0;
                    stop_bit = 3d0;            
                    end 
                        6,7,8,9,10,11:start_bit <= start_bit + s1_rs232_rx;
                        22,23,24,25,26,27:r_data_byte[0] <= r_data_byte[0] + s1_rs232_rx;
                        38,39,40,41,42,43:r_data_byte[1] <= r_data_byte[1] + s1_rs232_rx;
                        54,55,56,57,58,59:r_data_byte[2] <= r_data_byte[2] + s1_rs232_rx;
                        70,71,72,73,74,75:r_data_byte[3] <= r_data_byte[3] + s1_rs232_rx;
                        86,87,88,89,90,91:r_data_byte[4] <= r_data_byte[4] + s1_rs232_rx;
                        102,103,104,105,106,107:r_data_byte[5] <= r_data_byte[5] + s1_rs232_rx;
                        118,119,120,121,122,123:r_data_byte[6] <= r_data_byte[6] + s1_rs232_rx;
                        134,135,136,137,138,139:r_data_byte[7] <= r_data_byte[7] + s1_rs232_rx;
                        150,151,152,153,154,155:stop_bit <= stop_bit + s1_rs232_rx;
                    default:
                begin                                    
                    start_bit = start_bit;
                    r_data_byte[0] <= r_data_byte[0];
                    r_data_byte[1] <= r_data_byte[1];
                    r_data_byte[2] <= r_data_byte[2];
                    r_data_byte[3] <= r_data_byte[3];
                    r_data_byte[4] <= r_data_byte[4];
                    r_data_byte[5] <= r_data_byte[5];
                    r_data_byte[6] <= r_data_byte[6];
                    r_data_byte[7] <= r_data_byte[7];
                    stop_bit = stop_bit;                        
                end
        endcase
    end
                    
        
                                        always@(posedge clk or negedge rst_n)//数据提取
                                    if(!rst_n)
                                    data_byte <= 8d0;
                                else if(bps_cnt == 8d159)begin
                                    data_byte[0] <= r_data_byte[0][2];
                                    data_byte[1] <= r_data_byte [1][2];
                                    data_byte[2] <= r_data_byte[2][2];
                                    data_byte[3] <= r_data_byte[3][2];
                                    data_byte[4] <= r_data_byte[4][2];
                                    data_byte[5] <= r_data_byte[5][2];
                                    data_byte[6] <= r_data_byte[6][2];
                                    data_byte[7] <= r_data_byte[7][2];
                                end    
                                
                                //控制逻辑
                                
                        always@(posedge clk or negedge rst_n)
                    if(!rst_n)
                        uart_state <= 1b0;
                        else if(nedege)
                            uart_state <= 1b1;
                            else if(rx_done || (bps_cnt == 8d12 && (start_bit > 2)))    //接收完成或者到12位的时候,start_bit>2说明接收错误,起始位不对
                            uart_state <= 1b0;//关闭传输状态位
                        else
                            uart_state <= uart_state;        

endmodule

                    
                    
                    
                    

编写testbench并添加路径,因为testbench中调用了uart_rx,所以在添加路径的时候需要将uart.v文件添加到路径中去

`timescale 1ns/1ns
`define clock_period 20
module uart_rx_tb;
                        reg             rst_n;
                        reg             clk;
                        reg                rs232_rx;
                        wire            rs232_tx;
                        reg             [2:0]baud_set;
                        wire             rx_done;
                        wire             tx_done;
                        wire             [7:0]data_byte_r;
                        reg              [7:0]data_byte_t;
                        reg             send_en;
                      wire             uart_state;        
    
                        
                        

                uart_rx        uart_rx1(.clk(clk),
                                                    .rs232_rx(rs232_tx),    //用输入值作为读取值
                                                    .baud_set(baud_set),
                                                    .rst_n(rst_n),
                                                    .data_byte(data_byte_r),
                                                    .rx_done(rx_done)
                                                    );

                            uart_tx            uart_tx2(
                                                                                .clk(clk),
                                                                                .rst_n(rst_n),
                                                                                .send_en(send_en),
                                                                                .baud_set(baud_set),
                                                                                .tx_done(tx_done),
                                                                                .rs232_tx(rs232_tx),
                                                                                .data_byte(data_byte_t),
                                                                                .uart_state(uart_state)//将串口状态用LED显示出来
                                                                            );
                                            initial clk=1;
                                                                always#(`clock_period/2)  clk=~clk;
                                                                initial begin 
                                                                        rst_n<=1b0;
                                                                        data_byte_t<=8d0;
                                                                        send_en<=1d0;
                                                                        baud_set=3d4;
                                                                        #(`clock_period*20+1)
                                                                        rst_n<=1b1;
                                                                            #(`clock_period*50+1)
                                                                            data_byte_t<=8haa;
                                                                        send_en<=1d1;
                                                                        #(`clock_period)
                                                                        send_en<=1d0;
                                                                        @(posedge tx_done)//等待传输完成的上升沿
                                                                        #(`clock_period*5000)//重新发送
                                                                        data_byte_t<=8h55;
                                                                        send_en<=1d1;
                                                                        #(`clock_period)
                                                                        send_en<=1d0;
                                                                        #(`clock_period*500)
                                                                        $stop;
                                                                        end
endmodule 

技术分享

将uart_rx设为顶层文件,编译后进行仿真,仿真图形如下:

技术分享

新建ISSP IP核,添加8个探针

技术分享

 

在uart_top中将rs_232_rx添加进来

module uart_top(clk ,rst_n,rs232_rx,rs232_tx,key_in,led);

                    input key_in;
                    input clk;
                    input rst_n;
                    input  rs232_rx;
                    output rs232_tx;
                    wire send_en;
                    wire [7:0]data_byte_t;
                    reg [7:0]data_byte_r;
                    wire [7:0]data_rx;
                    output led;
                    wire key_flag,key_state;
                    wire rx_done;
                    assign send_en=key_flag&!key_state;//按键检测成功且为低电平时,发生使能
                    
                uart_tx            uart_tx1(
                                                            .clk(clk),
                                                            .rst_n(rst_n),
                                                            .send_en(send_en),
                                                           .baud_set(3d0),
                                                            .tx_done(),
                                                            .rs232_tx(rs232_tx),
                                                            .data_byte(data_byte_t),
                                                            .uart_state(led)//将串口状态用LED显示出来
                                                        );
                    uart_rx        uart_rx2(.clk(clk),
                                                    .rs232_rx(rs232_rx),    //用输入值作为读取值
                                                    .baud_set(3d0),
                                                    .rst_n(rst_n),
                                                    .data_byte(data_rx),//接收到的数据传递给data_rx;
                                                    .rx_done(rx_done)
                                                    );

                                key_filter key_filter0(
                                                                .clk(clk),
                                                                .rst_n(rst_n),
                                                                .key_in(key_in),
                                                                .key_flag(key_flag),
                                                                .key_state(key_state)
                                                                );
                    
                     ISSP     ISSP(
                                                            .probe(),//调用一个ISSP 发送数据
                                                            .source(data_byte_t)
                                                            );
                                 ISSP1  ISSP2(
                                                        .probe(data_byte_r),//调用一个ISSP 接收数据
                                                        .source()
                                                        );
                        
                                            //应为data_rx可能没有接收成功,为错误量,所以需要一个中间量来进行缓冲
                                        always@(posedge clk or negedge rst_n)
                                        if(!rst_n)
                                            data_byte_r <= 8d0;
                                        else if(rx_done)
                                            data_byte_r<= data_rx;
                                        else
                                            data_byte_r<= data_byte_r;
endmodule

 

 分配RX引脚给GPIO0_D0,然后将程序烧写到FPGA中,将GPIO0-D0与GPIO0_D1连着直接用杜邦线连接,打开ISSP工具,将两者都设置为hex显示,且将第二个ISSP工具设置为循环扫描,在第一个ISSP中输入数据,按下按键KEY1后,看到第二个ISSP 有相应的数据变化。

此时说明整个协议编写完成 了可以完成通讯了

技术分享

 

 技术分享

 

uart通讯协议