通信协议:Uart的Verilog实现(下)

news2025/1/22 21:41:52

4、UART接收器

        UART接收器负责接收串行比特流,去除起始位和停止位,并以并行格式将数据保存到与主机数据总线相连的寄存器里。接收器无法获得发送时钟,因此尽管数据以标准比特率到达,但数据未必与接收主机内的时钟同步。同步问题可以用同步器等方法解决,数据的采集可以使用一个更高频率的本地时钟进行采样。为保证采样是在比特时间的中间进行,应对Sample_clock时钟周期进行计数,如下图所示。采样方法必须保证:(1)能够检测到起始位到达;(2)能够采样到8个数据位;(3)能够把采样数据送到本地总线。

        虽然可以采用更高的采样频率,但本例中Sample_clock的频率定为(已知)发送时钟频率的8倍。这可以保证Sample_clock前沿与起始位之间的少许差异不会影响采样,因为只需在起始位对应的比特时间之内采样完成即可。具体来说,就是当串行输入被采样为低电平后表示起始位到来,而后将再进行三次采样增加可信度,在此后的连续数据位都将在比特时间的中间附近被采样(即控制两次采样间隔时间是比特时间),这是通过数据通路中的计数器来实现的。

        下图给出了接收器的内部框图,包括控制单元和数据通路,可以看到其中控制单元和主机之间的接口信号,已经控制单元和数据通路之间的控制信号与状态信号。

        控制单元的输入信号及其作用:

        read_not_ready_in:表示主机未准备好接收。

        Ser_in_0:当Serial_in = 0时有效,是由数据通路提供的状态信号。

        SC_eq_3:当Sample_counter = 3时有效,是由数据通路提供的状态信号。

        BC_er_8:当Sample_count < 7时有效,是由数据通路提供的状态信号。

        Sample_counter:对两次采样间隔进行计数。

        Bit_counter:计数已采样的数据位数。

        控制单元的输出信号及其作用:

        read_not_ready_out:表示接收机已接收到8位数据。

        clr_Sample_counter:控制数据通路的Sample_counter计数器清零。

        inc_Sample_counter:控制数据通路的Sample_counter计数值加1。

        clr_Bit_counter:控制数据通路的Bit_counters计数器清零。

        inc_Bit_counter:控制数据通路的Bit_counters计数值加1。

        shift:控制数据通路的RCV_shftreg向LSB方向移位。

        load:控制数据通路的RCV_shftreg数据传送到RCV_datareg。

        Error1:最后一个数据位采样结束后主机还没有准备好接收数据时有效。

        Error2:停止位丢失时有效。

        接收器控制器状态机ASMD如下图所示。该状态机包括idle、starting和receiving三个状态。状态之间的转移由Sample_clk来同步。低有效的同步复位输入rst_b使状态机进入idle状态,直到状态信号Ser_in_0变为高电平(串行输入为0)后状态机进入starting状态。在starting状态下,状态机重复采样Serial_in以确认是否是有效起始位。在Sample_clock的下一个有效沿,clr_Sample_counter和inc_Sample_counter需根据采样值确认是增加计数值还是清零:若接下来的连续三次采样值均为0,则认定为有效起始位,状态机转移到receiving状态并将给出clr_Sample_counter信号控制Sample_counter清零。在receiving状态下inc_Sample_counter将控制Sample_counter计数值增加以进行8个时钟周期的计时,对每个有效比特,在其比特时间的中间采样,总共采样7位数据位,1位校验位,Bit_counter增加。若采样的不是校验位,则inc_Bit_counter和shift持续有效。信号shift有效时,串行输入将载入接收器的移位寄存器RCV_shftreg的MSB位,且寄存器最左边的7位将向LSB方向移动。

module UART_RCVR #(parameter word_size = 8, half_word = word_size / 2)(
    output [word_size - 1 : 0]RCV_datareg,
    output read_not_ready_out,
           Error1, Error2,
    input  Serial_in,
           read_not_ready_in,
           Sample_clk,
           rst_b
);
    wire Ser_in_0, SC_eq_3, SC_lt_7, BC_eq_8,
         clr_Sample_counter, inc_Sample_counter,
         clr_Bit_counter, inc_Bit_counter,
         shift, load;
    

    Control_Unit M0(read_not_ready_out, Error1, Error2, clr_Sample_counter,
                    inc_Sample_counter, clr_Bit_counter, inc_Bit_counter,
                    shift, load, read_not_ready_in, Ser_in_0, SC_er_3,
                    SC_lt_7, BC_er_8, Sample_clk, rst_b);
    
    Datapath_Unit M1(RCV_datareg, Ser_in_0, SC_eq_3, SC_lt_7, BC_eq_8, Serial_in,
                     clr_Sample_counter, inc_Sample_counter, clr_Bit_counter,
                     inc_Bit_counter, shift, load, Sample_clk, rst_b);
    
endmodule

module Control_Unit #(parameter word_size = 8, half_word = word_size / 2,
                      Num_state_bits = 2)(
    output reg read_not_ready_out,
               Error1, Error2, clr_Sample_counter,
               inc_Sample_counter, clr_Bit_counter, inc_Bit_counter,
               shift, load
    input read_not_ready_in, Ser_in_0, SC_er_3,
          SC_lt_7, BC_er_8, Sample_clk, rst_b
);

    localparam idle = 2'b00,
               starting = 2'b01,
               receiving = 2'b10;
    reg [word_size - 1 : 0] RCV_shftreg;
    reg [Num_state_bits - 1 : 0] state, next_state;

    always@(posedge Sample_clk)
        if(rst_b == 0)
            state <=idle;
        else
            state <= next_state;

    always@(*)begin
        read_not_ready_out = 0;
        clr_Sample_counter = 0;
        clr_Bit_counter = 0;
        inc_Sample_counter = 0;
        inc_Bit_counter = 0;
        shift = 0;
        Error1 = 0;
        Error2 = 0;
        load = 0;
        next_state = idle;
        case(state)
            idle: if(Ser_in_0 == 1)
                      next_state = starting;
                  else
                      next_state = idle;
            starting: if(Ser_in_0 == 0)begin
                          next_state = idle;
                          clr_Sample_counter = 1;
                      end
                      else if(SC_eq_3 == 1)begin
                          next_state = receiving;
                          clr_Sample_counter = 1;
                      end
                      else begin
                          next_state = starting;
                          inc_Sample_counter = 1;                    
                      end
            receiving: if(SC_lt_7 == 1)begin
                           inc_Sample_conter = 1;
                           next_state = receiving;
                       end
                       else begin
                           clr_Sample_counter = 1;
                           if(!BC_eq_8)begin
                               next_state = receiving;                              
                               shift = 1;
                               inc_Bit_counter = 1;                               
                           end
                           else begin
                               next_state = idle;
                               read_not_ready_out = 1;
                               clr_Bit_counter = 1;
                               if(read_not_ready_in == 1) 
                                   Error1 = 1;
                               else if(Ser_in_0 == 1)
                                   Error2 = 1;
                               else
                                   Load = 1;
                           end
                       end
            default: next_state = idle;
        endcase 
    end
endmodule

module Datapath_Unit #(parameter word_size = 8, half_word = word_size / 2,
                       Num_counter_bits = 4)(
    output reg [word_size - 1 : 0] RCV_datareg,
    output Ser_in_0,SC_eq_3, SC_lt_7, BC_eq_8,
    input Serial_in, clr_Sample_counter, inc_Sample_counter, clr_Bit_counter,
          inc_Bit_counter, shift, load, Sample_clk, rst_b
);
    reg [word_size - 1 : 0] RCV_shftreg;
    reg [Num_counter_bits - 1 : 0] Sample_counter;
    reg [Number_counter_bits : 0] Bit_counter;

    assign Ser_in_0 = (Serial_in == 0);
    assign BC_eq_8 = (Bit_counter == word_size);
    assign SC_lt_7 = (Sample_counter < word_size - 1);
    assign SC_eq_3 = (Sample_counter == half_word - 1);

    always@(posedge Sample_clk)
        if(rst_b == 0)begin
            Sample_counter <= 0;
            Bit_counter <= 0;
            RCV_datareg <= 0;
            RCV_shftreg <= 0;
        end
        else begin
            if(clr_Sample_counter == 1)
                Sample_counter <= 0;
            else if(inc_Sample_counter == 1)
                Sample_counter <= Sample_counter + 1;
            if(clr_Bit_counter == 1)
                Bit_counter <= Bit_counter;
            else if(inc_Bit_counter == 1)
                Bit_counter <= Bit_counter + 1;
            if(shift == 1)
                RCV_shftreg <= {Serial_in, RCV_shftreg[word_size - 1 : 1]};
            else if(load == 1)
                RCV_datareg <= RCV_shftreg;
        end
endmodule

 以上内容来源于《Verilog HDL高级数字设计》,有删改

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1046134.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

增材云荣获2023世界制造业大会“安徽省重点工业互联网平台”称号

9月21日上午&#xff0c;2023世界制造业大会工业互联网专场发布会在合肥滨湖会展中心发布厅成功举办。会上发布了安徽省工业互联网领域的系列研究成果和创新应用案例。增材云平台深耕3D打印领域&#xff0c;整合3D打印产业链六大资源&#xff0c;以专业全面的技术助推行业快速发…

【lesson12】进程地址空间初识

文章目录 初识进程地址空间进程地址空间的具体分布和演示用户空间 VS 内核空间Linux VS Windows 初识进程地址空间 首先我们用代码演示一个问题大家思考一下。 #include <stdio.h>#include <unistd.h>int g_val 100;int main(){pid_t id fork();if(id 0){//子进…

ubuntu与win之间共享文件夹

ubuntu上设置共享文件夹 第一步&#xff1a;点击【设置】或【虚拟机弹窗下面的【设置】选项】 第二步&#xff1a;进入【虚拟机设置】页面&#xff0c;点击【选项】如下图所示 第三步&#xff1a;启用共享文件&#xff1a;点击【总是启用】第四步&#xff1a;添加共享文件&…

crypto:RSA

题目 利用代码跑一下解码 import gmpy2 e 17 p 473398607161 q 4511491 d gmpy2.invert(e,(p-1)*(q-1)) print(d)总结 RSA&#xff08;Rivest-Shamir-Adleman&#xff09;是一种非对称加密算法&#xff0c;常用于数据加密和数字签名。它基于两个大素数的乘积难以分解的数…

Java内存泄漏知识(软引用、弱引用等)

关于作者&#xff1a;CSDN内容合伙人、技术专家&#xff0c; 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 &#xff0c;擅长java后端、移动开发、商业变现、人工智能等&#xff0c;希望大家多多支持。 未经允许不得转载 目录 一、导读二、概览三、相关知识3.1 内存…

2009-2018年各省涉农贷款数据(wind)

2009-2018年各省涉农贷款数据&#xff08;wind&#xff09; 1、时间&#xff1a;:209-2018年 2、范围&#xff1a;31省 3、来源&#xff1a;wind 4、指标&#xff1a;涉农贷款 指标解释 &#xff1a;在涉农贷款的分类上&#xff0c;按照城乡地域将涉农贷款分为农村贷款和城…

【C语言】文件操作(一)

前言 本篇博客讲解对文件的操作&#xff0c;包括打开&#xff0c;关闭操作。在下篇博客将讲解文件的读写。 文章目录 一、 什么是文件&#xff1f;1.1 用于存储数据1.2 文件类型1.3 文件名1.4 二进制文件和文本文件 二、文件的打开和关闭2.1 流和标准流2.2 文件指针2.3文件的打…

软件测试行业痛点分析

做软件测试的同学们&#xff0c;你在平时的测试工作中有哪些困惑或困扰呢&#xff1f;你可以自行简单思考一下。下面我梳理一下&#xff0c;大家可以看看自己是不是也有如此的感受。 从测试整体角度分析&#xff1a; 第一个痛点是入门容易深入难。 很多人认为软件测试也就那么…

速冻品、预制菜商城小程序的作用有哪些

速冻品和预制菜也有很高的市场需求度&#xff0c;如外卖店、早餐速食快餐店等&#xff0c;可以大幅度降低人工操作时间及成本&#xff0c;除了产品批发外&#xff0c;比如速冻水饺等零售也有市场。 而随着预制菜/冷冻品深入市场&#xff0c;不少餐饮商家都会采购&#xff0c;对…

CV经典任务(二)目标检测 |单目标,多目标 非极大值抑制等

文章目录 1 目标检测1.1 单目标检测1.2 多目标检测3.2.1 阶段一 单像素点采样目标检测3.2.2 阶段二 多像素点采样目标检测3.2.3 阶段三 RNN3.2.4 阶段四 一阶段的目标检测 Yolo/SSD 1 目标检测 目标检测的重要任务是 目标定位&#xff1a;目标检测的首要任务是确定图像中对象…

(数组/字符串) 380. O(1) 时间插入、删除和获取随机元素 ——【Leetcode每日一题】

❓ 380. O(1) 时间插入、删除和获取随机元素 难度&#xff1a;中等 实现 RandomizedSet 类&#xff1a; RandomizedSet() 初始化 RandomizedSet 对象bool insert(int val) 当元素 val 不存在时&#xff0c;向集合中插入该项&#xff0c;并返回 true &#xff1b;否则&#x…

私房菜外卖配送商城小程序的作用是什么

私房菜有自己的品牌&#xff0c;也更容易跑出连锁经营体系&#xff0c;由于餐饮行业主要服务本地同城客户&#xff0c;因此在实际经营中对商家来说&#xff0c;怎样实现引流获客、促进到店及转化复购很重要。 传统线下传单、朋友圈推广等方式比较低效&#xff0c;而线上入驻第…

Spring Cloud Gateway实战WebFlux解析请求体及抛出指定错误代码和信息

概述 基于Spring Cloud开发微服务时&#xff0c;使用Spring Cloud原生自带的Gateway作为网关&#xff0c;所有请求都需要经过网关服务转发。 为了防止恶意请求刷取数据&#xff0c;对于业务请求需要进行拦截&#xff0c;故而可在网关服务增加拦截过滤器。基于此&#xff0c;有…

uniapp 事件委托失败 获取不到dataset

问题&#xff1a; v-for 多个span ,绑定点击事件 代码:view里包着一个span, <view class"status-list" tap"search"><span class"status-item" v-for"(key,index) in statusList" :key"index" :data-key"k…

程序员大佬‍被没收三年工资105w?

程序员大佬&#x1f468;&#x1f3fb;‍&#x1f4bb;被没收三年工资105w&#xff1f; 我在CSDN搜了一些关键词并没有找到关于这个事件的信息&#xff0c;发这篇文章主要是想让更多的大佬关注一下这个事情的发展&#xff0c;因此就放了几个图片&#xff0c;具体的大佬们可以去…

【短文】怎么查看自己的Linux是哪个发行版本和哪个版本号

2023年9月29&#xff0c;周三晚上 首先去到etc目录 cd /etc 然后用如下命令查看所有文件 ls 找到类似“XXX-release”的文件 然后用如下命令查看这个文件 cat

NodeMCU ESP8266基于Arduino IDE的开发环境搭建(图文并茂)

文章目录 NodeMCU ESP8266基于Arduino IDE的开发环境搭建&#xff08;手把手教程&#xff09;软件下载官网地址百度云 安装IDE配置基础配置设置开发板 测试串口驱动下载测试用例 总结 NodeMCU ESP8266基于Arduino IDE的开发环境搭建&#xff08;手把手教程&#xff09; 软件下…

力扣每日一题(+日常水几道题)

每日一题1333. 餐厅过滤器 - 力扣&#xff08;LeetCode&#xff09; 简单的按规则排序,去除几个不满足的条件然后排序返回即可 #include<algorithm> class Solution { public:vector<int> filterRestaurants(vector<vector<int>>& restaurants, …

涉及多条件查询 使用mybatispluse的解决方法EasyCaptcha图形验证码工具

登录中遇到账号和密码去数据库中查询因为查询是多条件的 所以需要使用QueryWrapper中allEq 而allEq如何添加条件使用map 位于mybatisplus的条件构造器的使用 条件构造器 | MyBatis-Plus QueryWrapper<User> wrapper new QueryWrapper<>();Map<String, Object&g…

rust所有权

一、堆和栈 栈和堆都是程序运行时使用的内存&#xff0c;但是它们的结构不同。 1.栈 栈&#xff0c;英文是stack。是内存的一段区域。 栈是后进先出形式的。就像薯片桶&#xff0c;先放进去的一片只能后拿出来。 栈上存储的数据大小必须是已知且固定的。也就是说如果一个变量…