STM32 串口Uart2频繁收发至其卡死的解决方案-基于正点原子库函数分析

一、项目描述:
需要STM32单片机与上位机使用串口频繁收发数据,收发频率为50Hz,即单片机每秒发送50帧数据,每秒接收50帧数据,每帧数据大概14字节。

二、问题描述:
收发一段时间后,单片机卡死,无反应。

三、前期知识:
本段引自《STM32F10x-中文参考手册》
3.1 知识点一
在这里插入图片描述 表180源自P538

由表180可知,当我们使能TXNEIE位时,不仅会开启接收中断TXNE,也会开启数据溢出中断ORE。因此,我们不仅要检测接收中断TXNE,也要检测ORE溢出位中断。

3.2 知识点二

在这里插入图片描述

源自P523

由红线内容所示,要复位ORE,则要先读SR再读DR。即先读状态寄存器(USART_SR),再读数据寄存器(USART_DR),即执行如下两行代码。

USART_ClearFlag(USART1,USART_FLAG_ORE);//清除ORE标志位
USART_ReceiveData(USART1);               //抛弃接收到的数据

四、代码部分
代码是基于STM32标准库编写。

4.1 源代码:

void USART1_IRQHandler(void)           //UART1中断服务函数
	{
	u8 Res;
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  
           //接收中断(接收到的数据必须以0x0d 0x0a结尾)
		{
		Res =USART_ReceiveData(USART1);	//读取接收到的数据
		
		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;//接收数据错误,重新开始接收	  
					}		 
				}
			} 
} 

4.2 原因分析
由上可知,

死机应为,未及时清除RXNE接收中断与ORE溢出中断标志位所致。

4.3 解决方案:
1.清除RXNE标志位与中断标志位

2.用USART_GetFlagStatus()函数检查是否发生ORE溢出中断,清除ORE中断标志位,抛弃所接受到的数据。

void USART1_IRQHandler(void)           //UART1中断服务函数
	{
	u8 Res;
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  
           //接收中断(接收到的数据必须以0x0d 0x0a结尾)
		{
		Res =USART_ReceiveData(USART1);	//读取接收到的数据
		
		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;//接收数据错误,重新开始接收	  
					}		 
				}
			} 
			
			//***************代码添加部分*************//
			USART_ClearFlag(USART1,USART_FLAG_RXNE);//清除RXNE标志位
			USART_ClearITPendingBit(USART1,USART_FLAG_RXNE);
                                                    //清除RXNE中断标志
     } 
		
		 	if(USART_GetFlagStatus(USART1, USART_IT_ORE) != RESET)  
                    //需要用USART_GetFlagStatus函数来检查ORE溢出中断
		{
			USART_ClearFlag(USART1,USART_FLAG_ORE);//清除ORE标志位
			USART_ReceiveData(USART1);	           //抛弃接收到的数据			
         } 
		  //****************代码添加结束****************//
} 

五、结果:
单片机收发正常,再无卡死现象。

原文链接:《STM32串口频繁收发至其卡死的解决方案》
参考链接:《STM32 串口接收大量数据导致死机》
参考链接:《STM32 串口发送导致死机 故障排除手记》

  • 12
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
你好!关于STM32收发数据包的问题,可以使用串口通信来实现。下面是一个简单的示例代码,演示了如何使用STM32串口收发数据包。 首先,你需要初始化串口。在STM32库函数中,可以使用HAL库来进行串口的初始化和配置。下面是一个初始化串口的示例代码: ```c UART_HandleTypeDef huart; void UART_Init(void) { huart.Instance = USART2; huart.Init.BaudRate = 115200; huart.Init.WordLength = UART_WORDLENGTH_8B; huart.Init.StopBits = UART_STOPBITS_1; huart.Init.Parity = UART_PARITY_NONE; huart.Init.Mode = UART_MODE_TX_RX; huart.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart.Init.OverSampling = UART_OVERSAMPLING_16; HAL_UART_Init(&huart); } ``` 接下来,你可以使用HAL库提供的函数来实现数据包的发送和接收。例如,使用`HAL_UART_Transmit()`函数发送数据包,使用`HAL_UART_Receive()`函数接收数据包。下面是一个简单的示例代码: ```c uint8_t txData[10] = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A}; uint8_t rxData[10]; void sendPacket(void) { HAL_UART_Transmit(&huart, txData, 10, 100); } void receivePacket(void) { HAL_UART_Receive(&huart, rxData, 10, 100); } ``` 以上代码演示了如何发送一个长度为10字节的数据包,并接收一个长度为10字节的数据包。 需要注意的是,以上只是一个简单的示例,实际应用中可能需要根据具体的需求进行修改和优化。同时,还需要根据具体的硬件连接和配置来进行相应的调整。 希望能对你有所帮助!如果还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ltqshs

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值