1. 开启串口,是能串口全局中断

  2. 配置DMA并勾选Memory选项

  3. 继续配置工程并且生成代码

  4. 添加一些串口通讯使用的全局变量

    #define BUFFER_SIZE 128
    uint8_t Tx_Buf[50] = {0};
    volatile uint8_t rx_len = 0; //接收数据长度
    volatile uint8_t recv_end_flag = 0; //接收完成标记位
    uint8_t rx1_buffer[BUFFER_SIZE]; //接收缓存
  5. 在主函数中开启串口中断和DMA接收

    __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);        // 使能idle中断
    HAL_UART_Receive_DMA(&huart1, Rx_Buf, BUFFER_SIZE); // 打开DMA接收

    注意:

    CubeMX生成的代码先初始化了串口再初始化DMA,会导致串口只能接收到最后一个字符的数据。这里需要将默认的初始化顺序调整一下,先初始化DMA,再初始化串口,即更改为:

    MX_DMA_Init();
    MX_USART1_UART_Init();

    参考:DMA初始化顺序问题

  6. 在中断处理函数中将串口数据存入缓冲区

    /**
    * @brief This function handles USART1 global interrupt.
    */
    void USART1_IRQHandler(void)
    {
    /* USER CODE BEGIN USART1_IRQn 0 */
    uint32_t tmp_flag = 0;
    uint32_t temp;
    tmp_flag = __HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE); //获取IDLE标志位
    if ((tmp_flag != RESET)) {
    // idle标志被置位
    __HAL_UART_CLEAR_IDLEFLAG(&huart1); //清除标志位
    temp = huart1.Instance->SR; //清除状态寄存器SR
    temp = huart1.Instance->DR; //读取数据寄存器中的数据
    HAL_UART_DMAStop(&huart1);
    temp = hdma_usart1_rx.Instance->NDTR; //获取DMA中未传输的数据个数
    rx_len = BUFFER_SIZE - temp; //得到已经接收的数据个数
    recv_end_flag = 1; //接受完成标志位置1
    } /* USER CODE END USART1_IRQn 0 */
    HAL_UART_IRQHandler(&huart1);
    /* USER CODE BEGIN USART1_IRQn 1 */ /* USER CODE END USART1_IRQn 1 */
    }
  7. 在主函数中轮询或在另一个线程中处理接收完成的数据

    if (recv_end_flag == 1) {
    printf("rx_len=%d\r\n", rx_len); //打印接收长度
    HAL_UART_Transmit(&huart1, Rx_Buf, rx_len, 200); //接收数据打印出来
    // TODO: 处理接收到的数据
    for (uint8_t i = 0; i < rx_len; i++) {
    Rx_Buf[i] = 0; //清接收缓存
    }
    rx_len = 0; //清除计数
    recv_end_flag = 0; //清除接收结束标志位
    HAL_UART_Receive_DMA(&huart1, Rx_Buf, BUFFER_SIZE); //重新打开DMA接收
    }
  8. 数据处理结束后要重新打开DMA

    HAL_UART_Receive_DMA(&huart1, Rx_Buf, BUFFER_SIZE); //重新打开DMA接收

STM32使用DMA接收不定长数据的更多相关文章

  1. STM32使用串口1配合DMA接收不定长数据,减轻CPU载荷

    STM32使用串口1配合DMA接收不定长数据,减轻CPU载荷 http://www.openedv.com/thread-63849-1-1.html 实现思路:采 用STM32F103的串口1,并配 ...

  2. STM32之串口DMA接收不定长数据

    STM32之串口DMA接收不定长数据 引言 在使用stm32或者其他单片机的时候,会经常使用到串口通讯,那么如何有效地接收数据呢?假如这段数据是不定长的有如何高效接收呢? 同学A:数据来了就会进入串口 ...

  3. 串口配合DMA接收不定长数据(空闲中断+DMA接收)-(转载)

    1.空闲中断和别的接收完成(一个字节)中断,发送完成(发送寄存器控)中断的一样是串口中断: 2.空闲中断是接收到一个数据以后,接收停顿超过一字节时间  认为桢收完,总线空闲中断是在检测到在接收数据后, ...

  4. 串口1配合DMA接收不定长数据(空闲中断+DMA接收)

    1.空闲中断和别的接收完成(一个字节)中断,发送完成(发送寄存器控)中断的一样是串口中断: 2.空闲中断是接收到一个数据以后,接收停顿超过一字节时间  认为桢收完,总线空闲中断是在检测到在接收数据后, ...

  5. STM32 HAL库使用中断实现串口接收不定长数据

    以前用DMA实现接收不定长数据,DMA的方法接收串口助手的数据,全部没问题,不过如果接收模块返回的数据,而这些数据如果包含回车换行的话就会停止接收,例如接收:AT\r\nOK\r\n,就只能接收到AT ...

  6. 关于socket客户端接收不定长数据的解决方案

    #!/usr/bin/env python3.5 # -*-coding:utf8-*- """ 本实例客户端用于不断接收不定长数据,存储到变量res "&qu ...

  7. STM32串口接收不定长数据原理与源程序(转)

    今天说一下STM32单片机的接收不定长度字节数据的方法.由于STM32单片机带IDLE中断,所以利用这个中断,可以接收不定长字节的数据,由于STM32属于ARM单片机,所以这篇文章的方法也适合其他的A ...

  8. Python3的tcp socket接收不定长数据包接收到的数据不全。

    Python Socket API参考出处:http://blog.csdn.net/xiangpingli/article/details/47706707 使用socket.recv(pack_l ...

  9. STM32 ~ USART接收不定长数据

    IDLE中断什么时候发生? IDLE就是串口收到一帧数据后,发生的中断.什么是一帧数据呢?比如说给单片机一次发来1个字节,或者一次发来8个字节,这些一次发来的数据,就称为一帧数据,也可以叫做一包数据. ...

  10. Stm32使用串口空闲中断,基于队列来接收不定长、不定时数据

    串口持续地接收不定长.不定时的数据,把每一帧数据缓存下来且灵活地利用内存空间,下面提供一种方式供参考.原理是利用串口空闲中断和DMA,每当对方发来一帧完整的数据后,串口接收开始空闲,触发中断,在中断处 ...

随机推荐

  1. div+css CSS基本

        • css 高度(height) • css 宽度(width)   · • css 边框(border)   · • css 背景(background)   · • css 浮动(floa ...

  2. Vue 收集表单数据-输入input,单选radio,多选checkbox,下拉框select ,以及v-model的3个修饰符(lazy,number,trim)

    From案例分析: 1.Html 部分: <form @submit.prevent="" style=" border: 1px solid rgb(109, 2 ...

  3. 前端电商 sku 的全排列算法

    需求 需求描述起来很简单,有这样三个数组: let names = ["iPhone",'iPhone xs'] let colors = ['黑色','白色'] let stor ...

  4. pyhon&QT编译

    1.编译qrcpyrcc5 -o ico_rc.py ./ico/ico.qrc.qrc文件格式<RCC> <qresource prefix="/"> & ...

  5. pytorch学习笔记(8)--搭建简单的神经网络以及Sequential的使用

    1.神经网络图 输入图像是3通道的32×32的,先后经过卷积层(5×5的卷积核).最大池化层(2×2的池化核).卷积层(5×5的卷积核).最大池化层(2×2的池化核).卷积层(5×5的卷积核).最大池 ...

  6. Ajax JavaScript传参-javascript加载本地文件

    get请求需要拼接到url里面 post请求,需要放到send方法里面,后端取值的时候,Ajax用JavaScript传输跟用postman传输,取值方式有些区别,这里做一下详细记录 传参格式{'id ...

  7. Thread 的run方法和start方法的区别

    start()方法是用来启动线程,实现了多线程运行 点进去查看源码,发现start方法创建一个线程 并让线程处于就绪状态,并且在start方法内会调用start0()方法,而start0作为本地方法  ...

  8. 重写react-navigation的stackNaviagtor产生的默认导航栏header样式

    主要是默认的stackNavigator产生的效果,很难看 重写这个阴影,在当前路由配置的 navigationOptions里的 headerStyle写样式 navigationOptions:{ ...

  9. Maven常用参数及其说明

    Maven常用参数及其说明 -h,--help Display help information-am,--also-make 构建指定模块,同时构建指定模块依赖的其他模块;-amd,--also-m ...

  10. Linux python后台任务

    Ubuntu 后台持续运行python服务 一般使用 nohup python -u app.py>t.log 2>&1 & nohup python3 -u app.py ...