当前位置:   article > 正文

STM32之串口中断接收丢失数据

STM32之串口中断接收丢失数据

五六年没搞STM32了,这个项目一切都挺顺利,万万没想到被串口接收中断恶心到了。遇到的问题很奇怪

 HAL_UART_Receive_IT(&huart1, &rx_buffer[rx_index], LCD_UART_LEN);

这个代码中 LCD_UART_LEN=1的时候,接收过来的数据,数据包的第一个字节总是会跑到rx_buffer的末尾,如下

当LCD_UART_LEN为其它值时,打印的看不出规律,有点乱,只有当LCD_UART_LEN=为数据包的大小时,打印出来没问题。

就这么一个万恶的破问题  高了一整天,参考了CubeMX库中的示例代码,一行行对照代码,不行!又参考了正点原子和野火的代码,这两家的代码可以说没有最烂,只有更烂,完全坑害初学者。比如正点原子的,竟然在中断服务中整个while循环判断状态。。。

最后没办法,抱着侥幸的心里试了试ChatGPT,给出了下面的示例代码

  1. #include "stm32f4xx_hal.h"
  2. #define BUFFER_SIZE 100
  3. uint8_t rx_buffer[BUFFER_SIZE];
  4. uint8_t rx_index = 0;
  5. UART_HandleTypeDef huart;
  6. void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
  7. {
  8. if (huart->Instance == USART1)
  9. {
  10. rx_buffer[rx_index++] = huart->Instance->DR; // 读取接收到的数据并放入缓冲区
  11. if (rx_index >= BUFFER_SIZE)
  12. {
  13. // 如果缓冲区已满,进行相应处理
  14. // 可以在这里进行数据处理或者清空缓冲区
  15. rx_index = 0; // 重置缓冲区索引
  16. }
  17. HAL_UART_Receive_IT(&huart, rx_buffer + rx_index, 1); // 继续接收下一个字节
  18. }
  19. }
  20. int main(void)
  21. {
  22. HAL_Init();
  23. __HAL_RCC_USART1_CLK_ENABLE();
  24. huart.Instance = USART1;
  25. huart.Init.BaudRate = 9600;
  26. huart.Init.WordLength = UART_WORDLENGTH_8B;
  27. huart.Init.StopBits = UART_STOPBITS_1;
  28. huart.Init.Parity = UART_PARITY_NONE;
  29. huart.Init.Mode = UART_MODE_TX_RX;
  30. huart.Init.HwFlowCtl = UART_HWCONTROL_NONE;
  31. huart.Init.OverSampling = UART_OVERSAMPLING_16;
  32. if (HAL_UART_Init(&huart) != HAL_OK)
  33. {
  34. // 初始化失败处理
  35. while (1);
  36. }
  37. HAL_UART_Receive_IT(&huart, rx_buffer, 1); // 启动中断接收
  38. while (1)
  39. {
  40. // 主循环中可以进行其他操作
  41. }
  42. }

最重要的一点就是读数据是直接寄存器读取的rx_buffer[rx_index++] = huart->Instance->DR;

或许这是HAL库的bug吧,我用的是STM32G0B0CET6,其它芯片的HAL库不清楚有没有这个问题。

就这么一个点 浪费了一整天。。。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小丑西瓜9/article/detail/457827
推荐阅读
相关标签
  

闽ICP备14008679号