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

博主五六年未搞STM32,项目中串口接收中断出现问题,当LCD_UART_LEN为1时,数据包首字节跑到rx_buffer末尾,其他值时数据无规律。参考示例代码无果,最后借助ChatGPT解决,怀疑是HAL库的bug,浪费了一整天时间。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

五六年没搞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,给出了下面的示例代码

#include "stm32f4xx_hal.h"

#define BUFFER_SIZE 100

uint8_t rx_buffer[BUFFER_SIZE];
uint8_t rx_index = 0;

UART_HandleTypeDef huart;

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
    if (huart->Instance == USART1)
    {
        rx_buffer[rx_index++] = huart->Instance->DR; // 读取接收到的数据并放入缓冲区
        if (rx_index >= BUFFER_SIZE)
        {
            // 如果缓冲区已满,进行相应处理
            // 可以在这里进行数据处理或者清空缓冲区
            rx_index = 0; // 重置缓冲区索引
        }
        HAL_UART_Receive_IT(&huart, rx_buffer + rx_index, 1); // 继续接收下一个字节
    }
}

int main(void)
{
    HAL_Init();

    __HAL_RCC_USART1_CLK_ENABLE();

    huart.Instance = USART1;
    huart.Init.BaudRate = 9600;
    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;
    if (HAL_UART_Init(&huart) != HAL_OK)
    {
        // 初始化失败处理
        while (1);
    }

    HAL_UART_Receive_IT(&huart, rx_buffer, 1); // 启动中断接收

    while (1)
    {
        // 主循环中可以进行其他操作
    }
}

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

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

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

### 实现STM32通过串口接收8字节数据包 为了实现STM32单片机通过串口中断正确接收并处理8个字节长度的数据帧,可以采用环形缓冲区技术来存储接收到的数据。这种方式能够有效防止数据丢失,并提高系统的实时性和稳定性。 #### 初始化配置 首先,在初始化阶段设置好串口参数以及使能相应的中断服务程序: ```c void USART_Init(void){ /* 配置GPIO */ /* 配置USART1 */ USART_InitStructure.USART_BaudRate = 9600; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No ; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART1, &USART_InitStructure); USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); //开启接收中断 NVIC_EnableIRQ(USART1_IRQn); //允许USART1中断请求 } ``` #### 中断服务例程 (ISR) 接下来定义一个用于响应接收事件的中断服务子程序(ISR),每当有新字符到达时触发此函数执行并将该字符存入到预先准备好的FIFO队列中去: ```c #define BUFFER_SIZE 64 // 定义缓冲区大小 uint8_t rxBuffer[BUFFER_SIZE]; // 接收缓存数组 volatile uint16_t head = 0; // 缓冲区头指针位置变量声明为易失型以防编译器优化错误 volatile uint16_t tail = 0; // 计算下一个索引值的方法宏定义 #define NEXT(index) (((index)+1)%BUFFER_SIZE) void USART1_IRQHandler(void) { if (USART_GetITStatus(USART1, USART_IT_RXNE) != RESET){ // 判断是否是RXNE标志位引起的中断 char ch = USART_ReceiveData(USART1); // 获取当前接收到的新字符 // 将其加入到循环缓冲区内 if(NEXT(head)!=tail){ // 如果不是满状态则继续操作 rxBuffer[head]=ch; head=NEXT(head); // 当接收到第8个字节时触发回调通知上层应用层逻辑进行进一步解析处理 static uint8_t byteCount=0; ++byteCount; if(byteCount==8){ ProcessReceivedFrame(); // 调用外部接口告知已完成整个消息体收集工作 byteCount=0; // 清零计数以便下次重新计算 } } USART_ClearITPendingBit(USART1, USART_IT_RXNE); //清除中断标记 } } // 处理完成后的回调方法 void ProcessReceivedFrame(){ // 此处放置对接收到的消息做具体分析和回应的操作... } ``` 上述代码片段展示了如何利用STM32内部资源构建起一套基于硬件中断驱动模式下的高效可靠的数据传输方案[^1]。值得注意的是,在实际编程过程中还需要考虑更多细节问题比如同步互斥访问共享资源等问题;另外对于不同版本固件库之间可能存在细微差异因此建议参照官方文档来进行适当调整[^2]。
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

冥焱破晓

开发不易,写文章更难,感谢鼓励

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

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

打赏作者

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

抵扣说明:

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

余额充值