STM32HAL库,解决串口UART中断接收到的第一个字节数据丢失

1.问题描述:

只有上电后第一次接收到的第一字节数据会丢失,往后再接收也不会存在问题了。

2.先贴出来重写UART中断回调函数

我在接收到第一字节数据后开启定时器中断的,做一个超时处理,每次接收到数据会对定时器计数值清零,如果超过6ms则认为一帧数据接收完毕。

c 复制代码
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
	
    if(huart->Instance == USART1)
    {
        if(uart1.rx_buf_cnt >= UART1_RXSIZE - 1)	//接收数据量超限,错误
        {
            uart1.rx_buf_cnt = 0;
            memset(uart1.rx_buf, 0x00, sizeof(uart1.rx_buf));		
            HAL_UART_Transmit(huart, (uint8_t *)"数据溢出", 10, 0xFFFF);
        }
        else									//接收正常
        {
            uart1.rx_buf[uart1.rx_buf_cnt] = RxBuffer;	//接收数据存储到rx_buf
			uart1.rx_buf_cnt++;
			if(uart1.rx_buf_cnt == 1)
			{			
				HAL_TIM_Base_Start_IT(&htim3);	//在接收到第一个数据后 开启定时器6ms中断	
			}			
            __HAL_TIM_SET_COUNTER(&htim3, 0);  //每次接收数据后清空计数器数值,防止超时
        }
        HAL_UART_Receive_IT(huart, (uint8_t *)&RxBuffer, 1); 
    }
}

上电第一次接收数据总是会少第一个字节。

3.解决过程

一开始以为是UART初始化后没有清除接收中断标志位,尝试过清除标志位,也根据网上说得的开中断要放在初始化后,但都没有效果,仿真过程中在中断回调函数中加入断点后就可以接收到,怀疑有什么影响到回调函数了,然后把开启定时器中断函数注释掉之后就可以了,就猜测是定时器初始化后中断标志位没有清零,开启后定时器中断后就直接进入中断函数把接收数据个数直接清零了,然后就从第二个数据重新开始接收。

3.解决最后

正如我的猜测,果然加入清除定时器中断标志位后数据就接收正常了。

原因:在定时器初始化的时候,为了更新预装载值,会置中断标志位,所以需要在NVIC开启前先清除一下标志位。

相关推荐
零一iTEM1 小时前
MAX98357A_音频输出测试
单片机·嵌入式硬件·开源·音视频·硬件工程
逐步前行1 小时前
STM32_标准库结构
stm32·单片机·嵌入式硬件
Hello_Embed1 小时前
libmodbus STM32 主机实验(USB 串口版)
笔记·stm32·学习·嵌入式·freertos·modbus
不做无法实现的梦~2 小时前
PX4各个模块的作用(3)
linux·stm32·嵌入式硬件·机器人·自动驾驶
清风6666662 小时前
基于单片机的喷漆机器人自动控制系统
单片机·嵌入式硬件·机器人·毕业设计·课程设计·期末大作业
不能跑的代码不是好代码2 小时前
STM32独立看门狗(IWDG)知识点及标准库使用指南
stm32·嵌入式硬件
程序员良许2 小时前
嵌入式处理器架构
后端·单片机·嵌入式
来自晴朗的明天2 小时前
15、隔离 DC-DC 电源电路(18S12 模块)
单片机·嵌入式硬件·硬件工程
我送炭你添花2 小时前
电子世界的奇妙冒险:从一个电阻开始(系列目录)
人工智能·单片机·嵌入式硬件·fpga开发
VekiSon3 小时前
Linux内核驱动——设备树原理与应用
linux·c语言·arm开发·嵌入式硬件