STM32串口发送中断试验、在初始化时不能使能串口发送中断

2023-05-16

问题描述

在测试STM32串口发送完成中断的应用中,遇到了一个很奇怪的问题,在初始化完成之后直接就进入了串口中断函数的发送完成服务中断函数部分。本测试代码是在原来的基础上更改的,原来只使能了接受中断,此次仅仅是在初始化中加入了使能发送完成中断,所以问题的重点就处在了这个使能发送完成中断了。

查阅资料发现,在使能了发送完成中断之后,硬件就会紧接着发送一个空字符,那么发送完成之后不就进入了中断服务函数了。一般的解决方法是在初始化中不使能发送完成中断。仅是在发送数据的函数中使能发送完成中断,然后在中断函数中Disable此中断。
在初始化中开启接收中断没大意义,因为必须在其接收中断服务函数中将其Disable掉,再在需要发送的函数中将其Enable

在初始化时,配置串口发送、接收引脚,配置串口参数,配置串口中断优先级。

切记:在初始化时不能使能串口发送中断,如果初始化时使能TXEIE中断允许,则会导致程序运行时一直在进串口中断。

初始化

void My_USART1_Init(void)
{
	  GPIO_InitTypeDef  GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);//使能USART1时钟
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
	
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource9,GPIO_AF_USART1);
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource10,GPIO_AF_USART1);

	  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
  GPIO_Init(GPIOA, &GPIO_InitStructure);
	
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
  GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	USART_InitStructure.USART_BaudRate=115200;
	USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode=USART_Mode_Rx|USART_Mode_Tx;
	USART_InitStructure.USART_Parity=USART_Parity_No;
	USART_InitStructure.USART_StopBits=USART_StopBits_1;
	USART_InitStructure.USART_WordLength=USART_WordLength_8b;
	
	USART_Init(USART1,&USART_InitStructure);
	USART_Cmd(USART1 ,ENABLE);
  
	USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);
	// // 不要在这里使能串口发送中断,否则一直进中断
	// USART_ITConfig(USART1, USART_IT_TC, ENABLE); 
	
	NVIC_InitStructure.NVIC_IRQChannel=USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;
	NVIC_Init(&NVIC_InitStructure);
	
	
}

编写中断函数

void USART1_IRQHandler(void)
{
	u8 res;
	if(USART_GetITStatus(USART1,USART_IT_RXNE)){
		
		res=USART_ReceiveData(USART1);
		USART_ITConfig(USART1,USART_IT_TC,ENABLE);
		USART_SendData(USART1,res);
	}
if(USART_GetITStatus(USART1, USART_IT_TC) != RESET)
	{
		USART_ITConfig(USART1, USART_IT_TC, DISABLE);
			USART_ClearFlag(USART1,USART_FLAG_TC);
	}
}

建议的是在初始化时不好启用TXE中断,只在要发送数据(尤其是字符串、数组这样的系列数据)时才启用TXE。在发送完成后立即将其关闭,以免引起不必要的麻烦。
对于发送,需要注意TXE和TC的差别——这里简单描述一下(具体参考https://editor.csdn.net/md/?articleId=129428646),假设串口数据寄存器是DR、串口移位寄存器是SR以及TXD引脚TXDpin,其关系是DR->SR->TXDpin。当DR中的数据转移到SR中时TXE置1,如果有数据写入DR时就能将TXE置0;如果SR中的数据全部通过TXDpin移出并且没有数据进入DR,则TC置1。并且需要注意TXE只能通过写DR来置0,不能直接将其清零,而TC可以直接将其写1清零。
对于发送单个字符可以考虑不用中断,直接以查询方式完成。
对于发送字符串/数组类的数据,唯一要考虑的是只在最后一个字符发送后关闭发送中断,这里可以分为两种情况:对于发送可显示的字符串,其用0x00作为结尾的,因此在ISR中就用0x00作为关闭发送中断(TXE或者TC)的条件;第二种情况就是发送二进制数据,那就是0x00~0xFF中间的任意数据,就不能用0x00来判断结束了,这时必须知道数据的具体长度。

参考文章:https://blog.csdn.net/weixin_43202477/article/details/84848295

实例-正点原子串口实验,发送由轮询改为中断

轮询

void USART1_IRQHandler(void)                	//串口1中断服务程序
{
	u8 Res;
#if SYSTEM_SUPPORT_OS 		//如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
	OSIntEnter();    
#endif
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)
	{
		Res =USART_ReceiveData(USART1);//(USART1->DR);	//读取接收到的数据
		
		if((USART_RX_STA&0x8000)==0)//接收未完成
		{
			if(USART_RX_STA&0x4000)//接收到了0x0d
			{
				if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始
				else USART_RX_STA|=0x8000;	//接收完成了 
			}
			else //还没收到0X0D
			{	
				if(Res==0x0d)USART_RX_STA|=0x4000;
				else
				{
					USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
					USART_RX_STA++;
					if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收	  
				}		 
			}
		}   		 
  } 
#if SYSTEM_SUPPORT_OS 	//如果SYSTEM_SUPPORT_OS为真,则需要支持OS.
	OSIntExit();  											 
#endif
	
} 
int main(void)
{ 
 
	u8 t;
	u8 len;	
	u16 times=0;  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
	delay_init(168);		//延时初始化 
	uart_init(115200);	//串口初始化波特率为115200
	LED_Init();		  		//初始化与LED连接的硬件接口
	while(1)
	{
		u8 i;
		if(USART_RX_STA&0x8000)
		{					   
			len=USART_RX_STA&0x3fff;//得到此次接收到的数据长度
			for(t=0;t<len;t++)
			{
				USART_SendData(USART1, USART_RX_BUF[t]);    //向串口1发送数据
				j=USART_GetFlagStatus(USART1,USART_FLAG_TC);
				while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);
			}
			printf("\r\n\r\n");//插入换行
			USART_RX_STA=0;
		
	}
		else
		{
		    times++;
			if(times%5000==0)
			{
				printf("\r\nALIENTEK 探索者STM32F407开发板 串口实验\r\n");
				printf("正点原子@ALIENTEK\r\n\r\n\r\n");
		    }
		if(times%200==0)printf("请输入数据,以回车键结束\r\n");  
			if(times%30==0)LED0=!LED0;//闪烁LED,提示系统正在运行.
			delay_ms(10);   
		}
	}
}

中断

void USART1_IRQHandler(void)                	//串口1中断服务程序
{
	u8 Res;
	u8 len;
	static u8 t;
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中断(接收到的数据必须是0x0d 0x0a结尾)
	{
		Res =USART_ReceiveData(USART1);//(USART1->DR);	//读取接收到的数据
		
		if((USART_RX_STA&0x8000)==0)//接收未完成
		{
			if(USART_RX_STA&0x4000)//接收到了0x0d,这是上次接受的数据,当接收到了0x0d,USART_RX_STA的bit14置1
			{
				if(Res!=0x0a)USART_RX_STA=0;//接收错误,重新开始,res是当前接收到的数据
				else {USART_RX_STA|=0x8000;	//接收完成了 
					USART_ITConfig(USART1,USART_IT_TC,ENABLE);
				}
			}
			else //还没收到0X0D,
			{	
				if(Res==0x0d)USART_RX_STA|=0x4000;
				else
				{
					USART_RX_BUF[USART_RX_STA&0X3FFF]=Res;//bit0-13的数据存放在USART_RX_BUF数组里
					USART_RX_STA++;
					if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收数据错误,重新开始接收,超过200了	  
				}		 
			}
		} 
		
  } 
	{
	if(USART_GetITStatus(USART1, USART_IT_TC) != RESET)
	{
			USART_SendData(USART1, USART_RX_BUF[t++]);         //向串口1发送数据
			while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET);//等待发送结束
			if(t == (USART_RX_STA&0X3FFF))//发送数据完成
			{  	
				t=0; 
				USART_RX_STA=0;
				USART_ITConfig(USART1, USART_IT_TC, DISABLE); //关闭发送中断
				//USART_ClearFlag(USART1,USART_FLAG_TC);//这里面不能清楚这个标志位,否则进不了发送中断,
				//USART_RX_STA无法清零,接收中断也近不了了,开始有点问题,换成TXE中断就好了
				printf("\r\n");//插入换行
			} 
			
	}
}

} 
int main(void)
{ 
 
	u8 t;
	u8 len;	
	u16 times=0;  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2
	delay_init(168);		//延时初始化 
	uart_init(115200);	//串口初始化波特率为115200
	LED_Init();		  		//初始化与LED连接的硬件接口  
	while(1)
	{
		times++;
			if(times%5000==0)
			{
				printf("\r\nALIENTEK 探索者STM32F407开发板 串口实验\r\n");
				printf("正点原子@ALIENTEK\r\n\r\n\r\n");
			}
			if(times%200==0)printf("请输入数据,以回车键结束\r\n");  
			if(times%30==0)LED0=!LED0;//闪烁LED,提示系统正在运行.
			delay_ms(10);
	}
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

STM32串口发送中断试验、在初始化时不能使能串口发送中断 的相关文章

  • 大疆 RoboMaster 3508/2006/GM6020 电机使用教程

    19年开始使用大疆的电机 xff0c 刚开始接触有很多东西不懂 xff0c 网上除了RM官网提供的一些资料外没有很多其他的资料 xff0c 现在使用大疆电机近一年了 xff0c 想分享一下自己的经验 1 硬件部分 1 C610电调只能连接M
  • DBC文件解析及CAN通信矩阵

    一般的 DBC 文件中包含了如下的8种信息 xff1a 1 版本与新符号 2 波特率定义 3 网络节点的定义 4 报文帧的定义 5 信号的定义 6 注解部分 7 特征部分 8 数值表部分 VERSOIN 34 34 版本信息 xff0c 为
  • 基于Rplidar二维雷达使用Hector_SLAM算法在ROS中建图

    文章目录 前言一 ROS分布式通信 xff08 配置多机通信 xff09 1 简介2 步骤2 1 准备2 2 修改配置文件2 3配置主机IP2 4配置从机IP 二 RPlidar的使用教程1 创建环境2 下载激光雷达的功能包3 编译4 启动
  • TCP连接建立的步骤

    TCP连接建立的步骤 一 客户端向服务器端发送连接请求后 xff0c 就被动地等待服务器的响应 典型的TCP客户端要经过下面三步操作 xff1a 1 创建一个Socket实例 xff1a 构造函数向指定的远程主机和端口建立一个TCP连接 x
  • 能否在头文件中放置函数定义?

    语法上是可以这样做的 xff0c 但是在编程规范中并不鼓励这样做 成员函数一般是不可以在头文件中定义的 xff0c 只能在头文件中声明 因为函数只能有一次定义 xff0c 而可以有多次声明 xff0c 当头文件被多次包含的时候 xff0c
  • 万能的sprintf

    0 前言 先推荐一本书 xff0c 政治书籍 政治的人生 xff0c 算是一本日记题材 是现任 xff0c 作者大家百度一下就知道了 xff0c 这里不宜过多说明 从这本书里 xff0c 可以看出来现在的社会 这本书是30年前的 大佬就是大
  • 串口通讯UART/RS232/RS485/RS-422笔记

    串口通讯详解笔记 串口通讯概述串口通讯传输数据帧的结构UARTRS232RS485RS 422RS 232 RS 422和RS 485的主要区别 xff08 重要 xff09 串口通讯概述 串口通讯是指数据按位 xff08 bit xff0
  • Stm32 hal库 usart2与hc-08透传模块通讯

    Stm32 hal库 usart2与hc 08透传模块通讯 xff08 附数据解析 xff09 一 stm32cubeMX配置 1 配置RCC为外部晶振 2 配置时钟树 3 配置usart1 usart2 xff0c 其中usart1将作为
  • darknet分类网络,训练,C++调用分类器

    Darknet 分类器 出于对Darknet框架下YOLO结构的火热 xff0c 网络上一堆关于目标检测的C 43 43 调用形式和模板 xff0c 但是未曾存在C 43 43 调用分类器的模板 xff0c 故采用如下形式 xff0c 展开
  • zed2 win10 采集数据

    环境 xff1a win10 cuda10 2 zed2相机 zed sdk 3 7 python3 7 1 标定 参考的博客 2 配置环境 1 xff09 win10安装cuda cudnn 如何查看windows的cuda版本 win1
  • 链表指针赋值

    总结来说 xff0c 就是等号赋值右边的指针 xff08 节点地址 xff09 不变 xff0c 左边的地址变成右边的 即左边的指针移到右边指针的位置 xff08 PS 指针命名时不要用next xff0c 会搞混 xff09 span c
  • 网络基础知识和常用数据帧格式

    网络基础知识和常用数据帧格式 1 IP路由相关1 1 网络分层1 2 网络分段1 3 子网掩码1 4 网关功能1 5 数据路由 2 常用帧格式2 1 ARP帧格式2 2 ICMP帧格式2 3 UDP帧格式2 4 TCP帧格式 本文主要介绍网
  • 解决github上的图片无法显示的问题

    把本地图片上传到 github 上后 xff0c 无法打开图片 xff0c 报如下图片 xff1a 百度了一下 xff0c 说什么DNS污染 xff0c 解决办法如下 xff1a 首先 xff0c 打开域名解析网站 xff1a https
  • ROS的ros_canopen调试

    Ros canopen是ros支持can通信的package 链接 xff1a http wiki ros org ros canopen distro 61 indigo Ros canopen包结构如下 SocketCAN是一组开源的C
  • HTTP请求携带用户密码验证

    在java操作es时 xff0c 可以使用HTTP请求的方式来连接es xff0c 一般es是没有密码限制的 但当es设置了用户名密码限制的时候我们需要在HTTP请求中携带用户名和密码 xff0c 如何在HTTP请求中携带用户名密码 xff
  • Java 使用Httpclient构建带有用户名与密码验证的get请求

    构造普通http请求 对于不需要用户名与密码认证的连接 xff0c 一般只需要构造好URIBuilder随后发起get请求即可获得返回信息 xff0c 具体代码如下 span class token class name Closeable
  • 全局变量和局部变量有什么区别?是怎么实现的?操作系统和编译器是怎么知道的?

    全局变量和局部变量有什么区别 xff1f 是怎么实现的 xff1f 操作系统和编译器是怎么知道的 xff1f
  • 推荐几本学习C#的书籍

    1 C 5 0 in a Nutshell 权威的参考书 学习C 的最佳书籍之一 如果你想要获取最新的信息以增长有关C 语言编程的知识 xff0c 这是最好的选择之一 该书深入浅出地探讨了有关C 语言的知识 不过要想能够充分吸收消化 xff
  • C++的类和C里面的struct有什么区别?

    C 43 43 的类和C里面的struct有什么区别 xff1f
  • linux使用curl进行接口传参

    curl 的命令是非常强大的 xff0c 这里主要介绍curl 43 post get相关的使用方式 1 PSOT 请求 引用文件 如果入参很长 xff0c 为了便于修改调整 xff0c 可以使用这种方式 xff1a span class

随机推荐

  • vscode 利用.clang-format 格式化C代码--连续宏定义和条件编译格式对齐

    最近在做项目时碰到一个需求 xff0c 将多个板子的头文件定义合并 xff0c 并整理其格式 xff0c 大概内容像下面这样 xff1a span class token macro property span class token di
  • 黏包现象和HTTP协议

    黏包现象和HTTP协议 黏包现象什么是黏包解决方案 HTTP协议HTTP协议概念URL的概念查看HTTP通信过程HTTP请求报文分析HTTP响应报文分析 静态web服务器什么是静态web服务器搭建python自带的静态web服务器搭建自己的
  • HTTP 使用GET方法 实现分包请求功能

    1 实现功能 从一个给定网页网址 xff08 比如 xff1a http www xxx com abcde bin xff09 下载bin文件 xff0c 并实现分包获取 bin文件内容 2 事先准备 4G模块 xff08 如 xff1a
  • 【STM32】UART 奇偶校验bug

    485设备的波特率固定为 9600bps xff0c 数据位为 8 位 xff0c 偶校验 xff0c 停止位为 1 位 stm32对应串口的配置为 xff1a huart4 Instance span class token operat
  • 在arduino上使用ESP8266模块编译出现No such file or directory解决办法

    首先确定在开发板管理器中已经安装好了ESP8266的包 xff0c 不会的可以网上搜一下 xff0c 或者在官方地址https github com esp8266 Arduino xff0c 向下找Installing with Boar
  • AxMath与mathtype的兼容问题

    如果是先安装了AxMath xff0c 后安装了mathtype xff0c 在word里进行公式编辑的时候会提示没有可用的公式编辑器 xff0c 文档里的公式就无法用任何方式打开了 解决方案 xff1a 卸载AxMath xff0c 再安
  • 使用BibLaTex编译参考文献时将论文集[C]后双斜杠//改为句号.

    使用了以下参考文献引用格式格式 xff1a span class token punctuation span addbibresource span class token punctuation span ref refs bib sp
  • 对‘cv Mat Mat()’未定义的引用‘cv imread(cv String const&, int)’未定义的引用 ...的解决办法

    对 cv Mat Mat 未定义的引用 cv imread cv String const amp int 未定义的引用 的解决办法 视觉SLAM十四讲的ch5的部分代码 xff0c 运行时出现了大面积的报错 xff0c 说相应的函数没有定
  • 打开多个相关联的网页 js脚本打开网页方法

    当日常工作中会有很多网页需要打开来使用 xff0c 如果每天重复的打开网页 xff0c 或者网页太多会很麻烦 那有没有其他的解决方法呢 xff1f 当然有 xff0c 下面用js的方法来解决这个问题 lt script gt var url
  • 栈的基本内容

    一 栈的定义 栈是限定仅在表尾进行插入和删除操作的线性表 通常把允许插入和删除的一端称为栈顶 xff0c 另一端称为栈底 xff0c 不含任何数据元素的栈称为空栈 栈又称为后进先出的线性表 xff0c 简称为LIFO结构 xff08 Las
  • 关于ERROR: error loading sources list: The read operation timed out 的解决方法

    在ubuntu 20 04系统下ROS平台安装时 xff0c 初始化rosdep报错 xff0c 在进行 rosdep update 出现超时报错 reading in sources list data from etc ros rosd
  • Http 解析url 各种参数类型工具类(文件,json,xml)

    1 http post请求传参数 43 文件 import java nio charset Charset import java util import org apache http HttpEntity import org apa
  • 一字节内的位序(bit)大端小端分析

    相信字节序 大端 小端的概念相关资料很多 xff0c 大家都比较清楚了 这里说明下一字节内部位序 xff08 bit xff09 的概念 在计算机中底层一个存储单元是字节 xff0c 因此你的指令寻址是不可能找到一字节内部的bit的 xff
  • RPLIDAR十分钟极速入门教程

    本教程由 臭皮匠机器人 原创 转载请注明出处 收到一个RPLidar后 xff0c 干的第一件事应该是先让雷达跑起来 xff0c 边做边学是最高效的 本教程 xff0c 将教你只用简单的三步 xff0c 花10分钟即可上手 本教程RPLid
  • RPLIDAR最强参数详解

    本教程由 臭皮匠机器人 原创 转载请注明出处 有的同学跟我们说 xff0c RPLIDAR的参数看不懂 xff0c 感觉理解起来困难 其实也不难懂的 xff0c 本篇将带你了解这些重要的指标 xff0c 给你讲清说透 由于RPLIDAR还在
  • 如何避免头文件被重复包含?

    在实际的工程中我们很有可能重复包含某一个头文件 xff0c 比如下面这种情况 xff1a 开发人员B和开发人员C在自己的头文件中都包含了开发人员A的头文件 xff0c 而开发人员D在自己的编译单元中包含了B和C的头文件 xff0c 此时D就
  • VScode配置C/C++编程总结(GCC+Clang+CMake)

    VScode配置C C 43 43 编程总结 xff08 GCC 43 Clang 43 CMake xff09 自己存下档纪念一下2019 12 18 Visual Studio Code VScode全称Visual Studio Co
  • 关于锂电池的二三事

    为了解释的更清楚 xff0c 我录制了一个视频 xff0c 详情可以参看 xff1a 我的这个B站视频 这是目录0 0 背景参数指标一些使用方法 背景 锂电池是我们日常生活中和制造东西时经常见到和使用的一种东西 xff0c 但是有很多细节可
  • 单片机堆栈知识总结

    堆栈 在片内RAM中 xff0c 常常要指定一个专门的区域来存放某些特别的数据 它遵循顺序存取和后进先出 LIFO FILO 的原则 xff0c 这个RAM区叫堆栈 其实堆栈就是单片机中的一些存储单元 xff0c 这些存储单元被指定保存一些
  • STM32串口发送中断试验、在初始化时不能使能串口发送中断

    问题描述 在测试STM32串口发送完成中断的应用中 xff0c 遇到了一个很奇怪的问题 xff0c 在初始化完成之后直接就进入了串口中断函数的发送完成服务中断函数部分 本测试代码是在原来的基础上更改的 xff0c 原来只使能了接受中断 xf