高云FPGA系列教程(8):ARM串口数据接收(中断和轮询方式)

文章目录

    • @[toc]
      • 1. GW1NSR-4C串口外设简介
      • 2. FPGA配置
      • 3. 常用函数
      • 4. 轮询方式接收数据
      • 5. 中断方式接收数据

本文是高云FPGA系列教程的第8篇文章。

本篇文章介绍片上ARM Cortex-M3硬核处理器串口外设的使用,演示轮询方式和中断方式接收串口数据,并进行回环测试,基于TangNano 4K开发板。

参考文档:Gowin_EMPU(GW1NS-4C)软件编程 参考手册

1. GW1NSR-4C串口外设简介

GW1NSR-4C ARM部分共有2个串口外设,都挂载在APB1总线上,最高支持波特率921.6Kbit/s,无奇偶校验位,8位数据位,1位停止位,支持高速测试模式 HSTM(High Speed Test Mode),即每个时钟周期输出1位数据,可以在短时间内传输大量数据。

官方手册上没有描述发送和接收缓存FIFO的深度,所以不确定是否支持缓存。

2. FPGA配置

FPGA部分需要在云源软件中手动使能EMPU串口外设,如下图所示。


不需要其他配置,使用起来非常简单。

3. 常用函数

高云串口驱动函数常用的有以下几个:

//串口初始化,指定波特率和中断使能,高速测试模式等
ErrorStatus UART_Init(UART_TypeDef* UARTx, UART_InitTypeDef* UART_InitStruct)
//获取接收缓存区状态,当接收到数据时,返回SET
FlagStatus UART_GetRxBufferFull(UART_TypeDef* UARTx)
//获取发送缓存区状态
FlagStatus UART_GetTxBufferFull(UART_TypeDef* UARTx)
//发送一个字节
void UART_SendChar(UART_TypeDef* UARTx,char txchar)
//发送字符串
void UART_SendString(UART_TypeDef* pUARTx, char *str)
//接收一个字节,轮询或接收中断时调用,自动
char UART_ReceiveChar(UART_TypeDef* UARTx)
//获取接收中断的状态,当被触发时返回SET
ITStatus UART_GetRxIRQStatus(UART_TypeDef* UARTx)
//获取发送中断的状态
ITStatus UART_GetTxIRQStatus(UART_TypeDef* UARTx)
//清除接收中断
void UART_ClearRxIRQ(UART_TypeDef* UARTx)
//清除发送中断
void UART_ClearTxIRQ(UART_TypeDef* UARTx)

下面来介绍串口接收数据的两种方式:轮询方式和中断方式。

4. 轮询方式接收数据

初始化时不使能接收中断:

void uart0_init(uint32_t BaudRate)
{UART_InitTypeDef UART_InitStruct;UART_InitStruct.UART_Mode.UARTMode_Tx = ENABLE;UART_InitStruct.UART_Mode.UARTMode_Rx = ENABLE;UART_InitStruct.UART_Int.UARTInt_Tx = DISABLE;UART_InitStruct.UART_Int.UARTInt_Rx = DISABLE;UART_InitStruct.UART_Ovr.UARTOvr_Tx = DISABLE;UART_InitStruct.UART_Ovr.UARTOvr_Rx = DISABLE;UART_InitStruct.UART_Hstm = DISABLE;UART_InitStruct.UART_BaudRate = BaudRate;//Baud RateUART_Init(UART0, &UART_InitStruct);
}

主循环中直接把收到的数据通过串口发送出去:

while(1)
{if(UART_GetRxBufferFull(UART0)){cnt_idle = 0;rx = UART_ReceiveChar(UART0);printf("rec data: %c\r\n", rx);}
}

这种简单粗暴的方式,会导致数据丢失,可能是串口接收部分没有FIFO导致:

我们可以采用缓冲区配合超时空闲的方式来处理,首先定义一个缓冲数组用来存储收到的数据,并通过一个计时器来判断当前是否空闲,若空闲则把数据返回:

uint8_t rx = 0;
uint8_t buf[256];
uint16_t buf_idx = 0;
uint32_t cnt_idle = 0;//空闲超时方式接收不丢失数据
while(1)
{//空闲时间计数器if(buf_idx != 0){cnt_idle++;}else {cnt_idle = 0;}//数据缓存到数组中if(UART_GetRxBufferFull(UART0)){cnt_idle = 0;buf[buf_idx] = UART_ReceiveChar(UART0);buf_idx++;}//长时间没有接收到串口数据,把缓冲区数据返回if(cnt_idle > 5000)   //明显感觉=500000{UART_SendString(UART0, buf);cnt_idle = 0;buf_idx = 0;memset(buf, 0, sizeof(buf)/sizeof(buf[0]));}
}

实际测试效果很不错,数据没有任何丢失:

下面来介绍通过串口接收中断的方式来缓存数据。

5. 中断方式接收数据

初始化时使能串口接收中断,并通过NVIC开启串口中断请求。

void uart0_init(uint32_t BaudRate)
{UART_InitTypeDef UART_InitStruct;NVIC_InitTypeDef InitTypeDef_NVIC;UART_InitStruct.UART_Mode.UARTMode_Tx = ENABLE;UART_InitStruct.UART_Mode.UARTMode_Rx = ENABLE;UART_InitStruct.UART_Int.UARTInt_Tx = DISABLE;UART_InitStruct.UART_Int.UARTInt_Rx = ENABLE;   //开启接收中断UART_InitStruct.UART_Ovr.UARTOvr_Tx = DISABLE;UART_InitStruct.UART_Ovr.UARTOvr_Rx = DISABLE;UART_InitStruct.UART_Hstm = DISABLE;UART_InitStruct.UART_BaudRate = BaudRate;//Baud RateUART_Init(UART0, &UART_InitStruct);//Enable UART0 interrupt handlerInitTypeDef_NVIC.NVIC_IRQChannel = UART0_IRQn;InitTypeDef_NVIC.NVIC_IRQChannelPreemptionPriority = 1;InitTypeDef_NVIC.NVIC_IRQChannelSubPriority = 1;InitTypeDef_NVIC.NVIC_IRQChannelCmd = ENABLE;NVIC_Init(&InitTypeDef_NVIC);
}

串口中断服务函数,数据缓存到数组中,并清零空闲计数器:

void UART0_Handler(void)
{char rx = 0;if(UART_GetRxIRQStatus(UART0) == SET){rx = UART_ReceiveChar(UART0);buf[buf_idx] = rx;buf_idx++;cnt_idle = 0;}UART_ClearRxIRQ(UART0);
}

需要注释掉系统默认提供的串口中断服务函数,否则编译会报错。

主循环中通过一个计数器来判断串口是否空闲,当超时没有收到新的数据时,认为串口空闲,把缓冲区的数据返回:

uint8_t rx = 0;
uint8_t buf[256];
uint16_t buf_idx = 0;
uint32_t cnt_idle = 0;while(1)
{//长时间没有接收到串口数据if(buf_idx != 0)cnt_idle++;else cnt_idle = 0;if(cnt_idle > 5000)   //明显感觉=500000{printf("rx: %s", buf);cnt_idle = 0;buf_idx = 0;memset(buf, 0, sizeof(buf)/sizeof(buf[0]));}
}

下载,运行,数据完整:


本文是高云FPGA系列教程的第8篇文章。

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

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

相关文章

安卓机型固件系统分区的基础组成 手机启动规律初步常识 各分区的基本含义与说明

此贴为基本常识。感兴趣的友友可以了解手机的启动顺序和各模式的基本操作与意义。另外了解手机系统分区各文件夹的含义 分区说明对应贴:安卓机型固件中分区对应说明 手机开机基本启动顺序 当我们按下手机开机键的时候。基本的启动顺序为 注意:该结构图…

Facebook最佳聊单工具--SaleSmartly,智能回复+控评+群控分流

关于SaleSmartlySaleSmartly--全渠道客户沟通平台,它可以帮助企业实现聊天自动化、智能化,提高员工效率,降低人工成本,提升客服质量。 在管理facebook时,你遇到的痛点: (1)FB聊单如何实现业务最…

大麦订单生成器最新版 大麦订单一键生成截图

1.可以一键添加,生成的假订单没有水印,界面也很真实。 2.在软件中输入生成的信息,这是产品信息,选择生成的产品图像,最后生成它。 后台一键生成,独立后台管理 教程:解压源码,修改…

用于无功补偿的固定电容晶闸管控制反应器研究(Simulink)

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…

03MyBatis-Plus中的常用注解

常用注解 TableName MyBatis-Plus根据BaseMapper中指定的泛型(实体类型名)确定数据库中操作的表,如果根据实体类型名找不到数据库中对应的表则会报表不存在异常 //向表中插入一条数据 Test public void testInsert(){User user new User(null, "张三", 23, "…

深入学习 Redis Sentinel - 基于 DockerCompose 编排哨兵分布式架构,理解工作原理

目录 一、哨兵模式 1.1、为何引入哨兵模式 1.2、Redis Sentinel 分布式架构 1.2.1、概述 1.2.2、工作原理(redis 哨兵的核心功能) 1. 监控: 2. 自动故障转移: 3. 通知 1.2.3、问题:哨兵结点只有一个可以么&am…

SpringCloud——微服务

微服务技术栈 在之前的开发过程中,我们将所有的服务都部署在一台服务器中,当我们的服务开始越来越多,业务越来越复杂,当一台服务器不能承担我们的业务的时候,就需要将不同的业务分开部署在不同的服务器上,…

Feign实战-Springboot集成OpenFeign Demo以及参数详解

最近整理一下微服务的文章,先拿一直用的OpenFeign开刀 思考:微服务之间如何方便优雅的实现服务间的远程调用 一、说说openFeign是什么吧? 说到这个,那不得不先说说RPC 1.什么是RPC RPC 全称是 Remote Procedure Call &#x…

Appium混合页面点击方法tap的使用

原生应用开发,是在Android、IOS等移动平台上利用官方提供的开发语言、开发类库、开发工具进行App开发;HTML5(h5)应用开发,是利用Web技术进行的App开发。目前,市面上很多app都是原生和h5混合开发&#xff0c…

竞赛选题 基于深度学习的中文情感分类 - 卷积神经网络 情感分类 情感分析 情感识别 评论情感分类

文章目录 1 前言2 情感文本分类2.1 参考论文2.2 输入层2.3 第一层卷积层:2.4 池化层:2.5 全连接softmax层:2.6 训练方案 3 实现3.1 sentence部分3.2 filters部分3.3 featuremaps部分3.4 1max部分3.5 concat1max部分3.6 关键代码 4 实现效果4.…

知识库系统推荐,强大的全文检索与文档分类管理功能

在我们日常企业运营管理过程中,会积累大量的文档资料,对于我们全体成员来说,这些知识文档都是巨大的财富,所以整合并搭建一套知识库系统是非常有必要的。 知识库系统推荐 我们日常工作中产生大量的文档,随着存储技术的…

pte初步认识学习

我们的时间的确很少,但是我们每天都乐意将珍贵的时间浪费在大量毫无意义的事情上 目录 pte介绍 PTE口语评分规则 pte架构 计算机科学23 QS排名 《芭比》 pte介绍 PTE口语评分规则 有抑扬顿挫 对于连读 不能回读 native pte对于个别单词没有读好&#xff0c…

【性能优化下】组织结构同步优化二,全量同步/增量同步,断点续传实现方式

看到这一篇文章的 xdm ,应该对组织结构同步有一些想法了吧,如果没有,可以看前面两篇文章,可以通过如下地址查看一下: 【性能优化上】第三方组织结构同步优化一,你 get 到了吗? 坑爹&#xff0c…

设计模式Java实战

文章目录 一、前置1.1 目的1.2 面向对象1.3 接口和抽象类 二、七大设计原则2.1 单一职责2.2 接口隔离原则2.3 依赖倒转原则2.4 里氏替换原则2.5 开闭原则2.6 不要重复原则2.7 迪米特最少知道法则 三、23种设计模式3.1创建型:创建对象3.1.1 单例模式定义最佳实践场景…

servlet中doGet方法无法读取body中的数据

servlet中doGet方法不支持读取body中的数据。

10万单词例句表单词句子ACCESS\EXCEL数据库

原本我以为《3万5千英语句子英语例句大全ACCESS数据库》例句已经够多了,没想到今天遇到一个10万条英语单词例句的数据,非常适合与单词词典进行关联学习,例句多了单词的用法以及句子的掌握都更有效率。 截图下方有显示“共有记录数”&#xff…

一台主机外接两台显示器

一台主机外接两台显示器 写在最前面双屏配置软件双屏跳转 写在最前面 在使用电脑时需要运行多个程序,时不时就要频繁的切换,很麻烦 但就能用双屏显示来解决这个问题,用一台主机控制,同时外接两台显示器并显示不同画面。 参考&a…

100G QSFP28 100km光模块最新解决方案

随着信息时代的到来,数据传输的速度和距离要求越来越高。目前,易天光通信发布了具有超低成本、可实现100G超长距离传输新方案——100G QSFP28 100km光模块,该方案是在100G ZR4 80km光模块上的全面升级。 一、产品概述 100G ZR4 100km是专为…

【探索Linux】—— 强大的命令行工具 P.9(进程地址空间)

阅读导航 前言一、内存空间分布二、什么是进程地址空间1. 概念2. 进程地址空间的组成 三、进程地址空间的设计原理1. 基本原理2. 虚拟地址空间 概念 大小和范围 作用 虚拟地址空间的优点 3. 页表 四、为什么要有地址空间五、总结温馨提示 前言 前面我们讲了C语言的基础知识&am…

【web开发】7、Django(2)

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 一、部门列表二、部门管理(增删改)三、用户管理过渡到modelform组件四、modelform实例:靓号操作五、自定义分页组件六、datepick…