热门标签 | HotTags
当前位置:  开发笔记 > 程序员 > 正文

13.基础实验(2)异步串口收发的实现

异步串口收发的实现一、实验内容实现串口接收来自串口调试助手发送的数据,然后直接通过自己实现的串口程序将数据发送出去二、系统框图三、设计分析从串口接收开始&#x
异步串口收发的实现

一、实验内容

实现串口接收来自串口调试助手发送的数据,然后直接通过自己实现的串口程序将数据发送出去

二、系统框图

三、设计分析

从串口接收开始,这个比较容易实现。参考链接串口通信协议

img

不考虑奇偶校验位,一个字节的串口的发送过程,包含1个低电平的起始位,一个高电平的停止位,中间8个数据位,且数据位是从低位发送到高位。只要保证前一刻为高电平,下一时刻为低电平,则可以确定开始串口开始接收数据,因为不发送数据时,即空闲状态下相应的电平为高电平。

从起始位开始,检测串口的数据的接收。延时一个半起始位的时间后,检测第一个数据位的电平,然后再延时一个数据位的时间,检测第二数据位,直到检测最后一个数据位,这时便检测到了整个字节的数据(停止位不考虑)。具体可以使用下面的时序图实现。

一是控制标志位的时序图。

image-20201230201432344

一个数据位占用的时钟周期数BAUD为时钟频率/波特率,计数到半个计数单元HALF = 1/2 BAUD时,half_baud_flag拉高一个时钟周期,baud_flag则在计算到BAUD计数时拉高一个时钟周期。baud_flag和half_baud_flag分别作为读取数据位和数据位计数的重要标志。

二是uart_rx实现的时序图。

image-20201230201558215

根据先前产生的baud_flag和half_baud_flag,分别实现计数和数据移位读取,当计数到8时,数据清零。同时只要计数baud_cnt为1-8时读取数据位,进行相应的移位,从高位向低位移动。最后,输出相应的结果。

uart_tx的时序图和uart_rx基本类似,这里不再进行分析。

四、实验步骤

根据波形编写相关代码,速度会非常快。将波特率和时钟频率作为参数,可以随时进行修改。

五、实际波形仿真

接收端功能实现之后,再实现发送端。

接收端功能调试

先使用modelsim进行仿真。从结果来看,获得的数据是正确的。

image-20210104214836236

再看看相关的计数,也是正确的。

image-20210104215028056

整体波形的设计也是正确的。

image-20210104215127080

再使用signal tap II实际查看结果。看是否真的获得了正确的数据。

首先rx的下降沿表示数据的开始,rx_cnt开始计数。

image-20210105093807401

然后baud_cnt和baud_flag的产生。

image-20210105093928949

最重要的是rx_data和rx_en的产生。

image-20210105094117008

发现rx_en信号没有正确的产生。其实这部分在modelsim仿真时问题就出现,只不过没有仔细地分析,这是大忌。

发现代码有问题,重新修改后,查看编译效果。

image-20210105094644435

如上图所示,刚好和数据数据同步。

至此,接收端功能实现成功。

发送端功能调试

直接使用signal tap ii查看结果。

发现串口并没有接收到正确的数据,为什么呢。

image-20210105113406500

查看波形查找问题。

image-20210105113501342

发现tx开始数据时,并不是从高电平到低电平的跳变。问题并不是出在这,后面的时序中存在高电平到低电平的跳变也没有数据输出。

image-20210105125725216

发现问题出在没有进行相关的引脚配置,导致出现了问题。

整体联调

发现部分数据出现了错误。

image-20210105133047177

再用ila抓取数据,发现是接收端的问题。

主要是接收端的数据来自外部时钟,存在时序违例,需要打俩拍后,才开始接收相关数据,否则就会出现问题。

always @(posedge clk)
beginrx_r <&#61; {rx_r[1:0], rx};
end

问题解决&#xff0c;搞定了。

六、总结与讨论*

1.外部时钟信号的输入时&#xff0c;打俩拍解决时序违例的问题。这点需要牢记于心&#xff0c;不能瞎搞。


推荐阅读
  • Linux内核中的内存反碎片技术解析
    本文深入探讨了Linux内核中实现的内存反碎片技术,包括其历史发展、关键概念如虚拟可移动区域以及具体的内存碎片整理策略。旨在为开发者提供全面的技术理解。 ... [详细]
  • 本文探讨了如何选择一个合适的序列化版本ID(serialVersionUID),包括使用生成器还是简单的整数,以及在不同情况下应如何处理序列化版本ID。 ... [详细]
  • 本文详细探讨了PHP中使用const和define定义常量的方法及其差异。了解这些区别有助于开发者根据具体需求选择合适的方式定义常量。 ... [详细]
  • 在VSCode中配置ESP-IDF开发环境
    本文详细介绍如何在无需单独下载工具链的情况下,通过VSCode配置ESP-IDF开发环境,包括下载离线安装包和安装ESP-IDF插件,从而实现项目编译与烧录。 ... [详细]
  • 本文旨在探讨Swift中的Closure与Objective-C中的Block之间的区别与联系,通过定义、使用方式以及外部变量捕获等方面的比较,帮助开发者更好地理解这两种机制的特点及应用场景。 ... [详细]
  • Gradle 是 Android Studio 中默认的构建工具,了解其基本配置对于开发效率的提升至关重要。本文将详细介绍如何在 Gradle 中定义和使用共享变量,以确保项目的一致性和可维护性。 ... [详细]
  • C/C++中一级指针的内存模型与示例代码
    本文通过一个具体的C/C++代码示例,详细解析了一级指针在内存中的布局和工作原理。包括了对不同类型的指针变量如何在内存中分配空间的讨论。 ... [详细]
  • 本文探讨了Linux环境下线程私有数据(Thread-Specific Data, TSD)的概念及其重要性,介绍了如何通过TSD技术避免多线程间全局变量冲突的问题,并提供了具体的实现方法和示例代码。 ... [详细]
  • C/C++ 应用程序的安装与卸载解决方案
    本文介绍了如何使用Inno Setup来创建C/C++应用程序的安装程序,包括自动检测并安装所需的运行库,确保应用能够顺利安装和卸载。 ... [详细]
  • 本报告记录了嵌入式软件设计课程中的第二次实验,主要探讨了使用KEIL V5开发环境和ST固件库进行GPIO控制及按键响应编程的方法。通过实际操作,加深了对嵌入式系统硬件接口编程的理解。 ... [详细]
  • 本文分享了作者在使用LaTeX过程中的几点心得,涵盖了从文档编辑、代码高亮、图形绘制到3D模型展示等多个方面的内容。适合希望深入了解LaTeX高级功能的用户。 ... [详细]
  • 本文介绍了Tomcat的基本操作,包括启动、关闭及首次访问的方法,并详细讲解了如何在IDEA中创建Web项目,配置Servlet及其映射,以及如何将项目部署到Tomcat。 ... [详细]
  • IA64架构下常见编程陷阱探讨
    本文深入探讨了IA64架构中常见的一个编程错误案例,该案例揭示了当开发者试图绕过编译器的某些限制时可能遇到的问题。通过具体分析IA64架构的特点及其对全局变量处理的方式,本文旨在为开发者提供避免此类问题的有效建议。 ... [详细]
  • SSE图像算法优化系列三:超高速导向滤波实现过程纪要(欢迎挑战)
    自从何凯明提出导向滤波后,因为其算法的简单性和有效性,该算法得到了广泛的应用,以至于新版的matlab都将其作为标准自带的函数之一了&#x ... [详细]
  • 本文详细介绍了 Node.js 中 OS 模块的 arch 方法,包括其功能、语法、参数以及返回值,并提供了具体的使用示例。 ... [详细]
author-avatar
卜弃miao_286
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有