当前位置: 首页 > article >正文

处理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);
}

http://www.kler.cn/a/540729.html

相关文章:

  • Linux 安装 Ollama
  • 前端开发中,如何判断一个元素是否在可视区域中?
  • Expo运行模拟器失败错误解决(xcrun simctl )
  • 【MySQL】深度学习数据库开发技术:使用CC++语言访问数据库
  • 第一财经对话东土科技 | 探索工业科技新边界
  • sqlite 查看表结构
  • 中央处理器
  • 分布式微服务接口基于多线程进行性能优化
  • 蓝桥杯试题:冒泡排序 选择排序
  • 六.logback记录日志文件并按大小日期分割文件
  • 操作系统调度算法解析(SJF)
  • EtherNet/IP转Modbus TCP实现三菱变频器与西门子PLC通讯的配置案例
  • 从零复现DeepSeek R1:从V3中对MoE、MLA、MTP的实现,到Open R1对R1中SFT、GRPO的实现
  • ESP8266配置为TCP客户端,连接电脑和手机(使用Arduino配置)
  • javaEE-10.CSS入门
  • 【Elasticsearch】管道聚合
  • SpringBoot源码解析(十):应用上下文AnnotationConfigServletWebServerApplicationContext构造方法
  • 相对收益-固定收益组合归因-前言
  • 纯前度(vue)实现对pdf\mp4\png\jpg\jpegxls\doc\txt文件预览,无需要转化
  • Android图片加载框架Coil,Kotlin
  • 国税局发票查验中英文验证码识别,识别率99.99%
  • 十四. Redis 新功能
  • day10-字符串
  • Spring Boot 与 Kafka 实现高效消息队列通信的最佳实践
  • flutter ListView Item复用源码解析
  • 亲自测试并推荐的第三方平台使用DeepSeek 和 LM Studio 本地部署DeepSeek