1.问题描述:
只有上电后第一次接收到的第一字节数据会丢失,往后再接收也不会存在问题了。
2.先贴出来重写UART中断回调函数
我在接收到第一字节数据后开启定时器中断的,做一个超时处理,每次接收到数据会对定时器计数值清零,如果超过6ms则认为一帧数据接收完毕。
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_bufuart1.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开启前先清除一下标志位。