处理STM32 DMA方式下的HAL_UART_ERROR_ORE错误

在STM32微控制器中,使用DMA(直接存储器访问)方式进行UART通信时,可能会遇到 HAL_UART_ERROR_ORE(Overrun Error)错误。这种错误通常发生在接收缓冲区满时,新数据到来导致数据溢出。处理这种错误需要系统化的策略,包括正确配置DMA、适当的错误处理和数据管理。以下是详细的解决方法和步骤。

一、理解Overrun Error (ORE)

Overrun Error (ORE) :当USART接收器在缓冲区已满的情况下接收到新数据时,会产生溢出错误。此错误表示上一次接收的数据未及时处理,新数据已经到达,但无法存储。

二、配置DMA和UART

正确配置DMA和UART对于避免溢出错误至关重要。

1. 配置UART

在CubeMX中配置UART,确保启用DMA接收模式。

2. 配置DMA

配置DMA通道,确保DMA的优先级和缓冲区大小足够应对预期的数据速率。

三、编写代码处理ORE错误

1. 初始化UART和DMA

确保正确初始化UART和DMA。

// UART句柄定义
UART_HandleTypeDef huart1;
DMA_HandleTypeDef hdma_usart1_rx;

// UART和DMA初始化函数
void MX_USART1_UART_Init(void) {
    huart1.Instance = USART1;
    huart1.Init.BaudRate = 115200;
    huart1.Init.WordLength = UART_WORDLENGTH_8B;
    huart1.Init.StopBits = UART_STOPBITS_1;
    huart1.Init.Parity = UART_PARITY_NONE;
    huart1.Init.Mode = UART_MODE_TX_RX;
    huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
    huart1.Init.OverSampling = UART_OVERSAMPLING_16;
    if (HAL_UART_Init(&huart1) != HAL_OK) {
        Error_Handler();
    }
}

void MX_DMA_Init(void) {
    // DMA控制器时钟启用
    __HAL_RCC_DMA1_CLK_ENABLE();

    // USART1 DMA RX初始化
    hdma_usart1_rx.Instance = DMA1_Channel5;
    hdma_usart1_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;
    hdma_usart1_rx.Init.PeriphInc = DMA_PINC_DISABLE;
    hdma_usart1_rx.Init.MemInc = DMA_MINC_ENABLE;
    hdma_usart1_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;
    hdma_usart1_rx.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;
    hdma_usart1_rx.Init.Mode = DMA_CIRCULAR;
    hdma_usart1_rx.Init.Priority = DMA_PRIORITY_HIGH;
    if (HAL_DMA_Init(&hdma_usart1_rx) != HAL_OK) {
        Error_Handler();
    }

    __HAL_LINKDMA(&huart1, hdmarx, hdma_usart1_rx);
}
​

2. 启动DMA接收

在主程序中启动DMA接收:

uint8_t RxBuffer[64]; // 接收缓冲区

int main(void) {
    HAL_Init();
    SystemClock_Config();
    MX_GPIO_Init();
    MX_DMA_Init();
    MX_USART1_UART_Init();

    if (HAL_UART_Receive_DMA(&huart1, RxBuffer, sizeof(RxBuffer)) != HAL_OK) {
        Error_Handler();
    }

    while (1) {
        // 主循环
    }
}
​

3. 错误处理回调函数

实现UART错误回调函数,处理 HAL_UART_ERROR_ORE错误。

void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart) {
    if (huart->ErrorCode & HAL_UART_ERROR_ORE) {
        __HAL_UART_CLEAR_OREFLAG(huart); // 清除ORE错误标志

        // 重新启动DMA接收
        HAL_UART_Receive_DMA(huart, RxBuffer, sizeof(RxBuffer));
    }
}
​

四、优化数据处理和缓冲区管理

1. 提高数据处理速度

确保在接收数据后尽快处理,以避免缓冲区溢出。

2. 增加缓冲区大小

根据实际应用需求,增加DMA缓冲区大小。

3. 使用循环缓冲区

实现循环缓冲区,提高数据处理效率。

#define BUFFER_SIZE 256
uint8_t CircularBuffer[BUFFER_SIZE];
volatile uint16_t WriteIndex = 0;
volatile uint16_t ReadIndex = 0;

void ProcessData(void) {
    while (ReadIndex != WriteIndex) {
        uint8_t data = CircularBuffer[ReadIndex++];
        ReadIndex %= BUFFER_SIZE;

        // 处理数据
    }
}

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart) {
    // 将接收到的数据存入循环缓冲区
    CircularBuffer[WriteIndex++] = RxBuffer[0];
    WriteIndex %= BUFFER_SIZE;

    // 继续接收数据
    HAL_UART_Receive_DMA(huart, RxBuffer, 1);
}
相关推荐
聪聪060610 分钟前
使用STM32F103C8T6和ESP8266链接阿里云
stm32·嵌入式硬件·阿里云
鹿屿二向箔12 分钟前
单片机上SPI和IIC的区别
单片机·嵌入式硬件
weixin_580382061 小时前
STC51 P0 口 与P1 口输出
单片机·嵌入式硬件
爱吃奶酪的松鼠丶2 小时前
51单片机之使用Keil uVision5创建工程以及使用stc-isp进行程序烧录步骤
嵌入式硬件·51单片机·接口隔离原则
一只搬砖的猹3 小时前
PCA9685舵机控制板使用
stm32·单片机·嵌入式硬件·mcu·pwm·舵机·舵机驱动
零下273°4 小时前
51单片机俄罗斯方块开机动画
单片机·嵌入式硬件·51单片机
浅陌pa4 小时前
04:定时器
c语言·单片机·嵌入式硬件·51单片机
╰⋛⋋⊱⋋翅膀⋌⊰⋌⋚╯5 小时前
STM32-启动文件
stm32·单片机
爱吃奶酪的松鼠丶7 小时前
51单片机之引脚图(详解)
单片机·嵌入式硬件·51单片机
fengyuzhe139 小时前
STC 51单片机62——极简 4x4x4光立方
单片机·嵌入式硬件·51单片机