• STM32F407 串口使用DMA方式通信


    DMA的原理,就是利用寄存器方式进行读写,这样的好处就是相对于中断触发(往往一个字节字节的就中断一次),CPU中断次数大大降少,提高了效率,但也影响了实时性。总体来说,对于一般的应用,瑕不掩瑜,值得使用。

    本文是基于串口1的,实际上串口1也是printf重定向接口,貌似没有什么冲突。

    原则上:

    1. 串口接收采用DMA+空闲中断的方式

    2. 串口发送就是直接发送方式

    这样的方式,符合一般项目需求。

    也分两部分:STM32CubeMx端配置+代码的处理

    STM32CubeMx端配置

    CubeMX 这边在原先“串口”配置基础上,配置DMA方式,总体来说,跟网上大部分例子差不多。

    在USART界面下选择 DMA Settings,如下图:

    Mode选择Normal,一般都是。还有一种循环发送,大概意思就是开启后自动循环发送?

    第4步骤,根据实际情况下选择,貌似没啥影响

    RX:接收端配置

    TX:发送端配置

    在NVIC Settings。确认下是否中断开启?

    Parameters Settings 跟常规串口配置一样,主要是一些波特率参数之类的

    还要配置下DMA,在 System Core地方选择DMA

    最后一项MEMTOMEM,默认是没有的,可通过下面的Add添加上去

    其他都跟往常一样。

    代码侧改动

    好几个地方要改动:

    直接上代码:

    usart.h

    1. /* USER CODE BEGIN Private defines */
    2. /* xxxx 的意思是 你自己取个好点名字*/
    3. #define XXXX_BUFFER_SIZE 128
    4. extern volatile uint8_t g_xxxxRxLen; /*接收一帧数据的长度*/
    5. extern volatile uint8_t g_xxxxRecvEndFlag; /*一帧数据接收完成标志*/
    6. extern uint8_t g_xxxxRxbuffer[XXXX_BUFFER_SIZE]; /*接收数据缓存数组*/
    7. /* USER CODE END Private defines */

    usart.c

    1. /* USER CODE BEGIN 0 */
    2. volatile uint8_t g_xxxxRxLen = 0;
    3. volatile uint8_t g_xxxxRecvEndFlag = 0;
    4. uint8_t g_xxxxRxbuffer[XXXX_BUFFER_SIZE] = {0};
    5. /* USER CODE END 0 */
    6. void MX_USART1_UART_Init(void)
    7. 后面追加
    8. /* USER CODE BEGIN USART1_Init 2 */
    9. __HAL_UART_ENABLE_IT(&huart1, UART_IT_RXNE);
    10. __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE);
    11. /*DMA接收函数,此句一定要加,不加接收不到第一次传进来的实数据,*/
    12. /*是空的,且此时接收到的数据长度为缓存器的数据长度*/
    13. HAL_UART_Receive_DMA(&huart1,g_xxxxRxbuffer,XXXX_BUFFER_SIZE);
    14. /* USER CODE END USART1_Init 2 */

    stm32f4xx_it.c  增加以下内容(10x系列,就是f1xx)

    1. void USART1_IRQHandler(void)
    2. {
    3. /* USER CODE BEGIN USART1_IRQn 0 */
    4. uint32_t temp;
    5. if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET)
    6. {
    7. 可能是点个灯之类操作
    8. __HAL_UART_CLEAR_IDLEFLAG(&huart1);
    9. HAL_UART_DMAStop(&huart1);
    10. /* 获取DMA中未传输的数据个数 */
    11. temp = __HAL_DMA_GET_COUNTER(&hdma_usart1_rx);
    12. /*总计数减去未传输的数据个数,得到已经接收的数据个数 */
    13. /* 这里有风险的,BUFFER_SIZE 如果小于当前帧的话,就有问题*/
    14. /* 正常情况下,要考虑下此情况 */
    15. g_xxxxRxLen = XXXX_BUFFER_SIZE - temp;
    16. g_xxxxRecvEndFlag = 1; /* 标志位置1 */
    17. }
    18. /* USER CODE END USART1_IRQn 0 */
    19. HAL_UART_IRQHandler(&huart1);
    20. /* USER CODE BEGIN USART1_IRQn 1 */
    21. /* USER CODE END USART1_IRQn 1 */
    22. }

    应用侧,需要一个task去轮询(是否收到数据)

    比如说,我是放在Task3里面

    1. void StartTask03(void const * argument)
    2. {
    3. /* USER CODE BEGIN StartTask03 */
    4. /* Infinite loop */
    5. uint8_t dmaSend[] = "this is DMA\n";
    6. for(;;)
    7. {
    8. if(g_xxxxRecvEndFlag == 1)
    9. {
    10. /* 此demo 简单处理,就应答一个固定内容,实际要根据实际情况处理*/
    11. DMA_Usart1_Send(dmaSend,sizeof(dmaSend)-1);
    12. // 此处增加协议处理之类的。可以复杂
    13. g_xxxxRxLen = 0;
    14. g_xxxxRecvEndFlag = 0;
    15. memset(g_xxxxRxbuffer,0,g_xxxxRxLen);
    16. }
    17. HAL_UART_Receive_DMA(&huart1,g_xxxxRxbuffer,XXXX_BUFFER_SIZE);
    18. osDelay(1);
    19. }
    20. /* USER CODE END StartTask03 */
    21. }
    1. /*
    2. *********************************************************************************************************
    3. * 函 数 名: DMA_Usart_Send
    4. * 功能说明: 串口发送功能函数
    5. * 形 参: buf,len
    6. * 返 回 值: 无
    7. *********************************************************************************************************
    8. */
    9. void DMA_Usart1_Send(uint8_t *buf,uint8_t len)
    10. {
    11. while(HAL_DMA_GetState(&hdma_usart1_tx) == HAL_DMA_STATE_BUSY)
    12. {
    13. osDelay(1);
    14. }
    15. /* 关闭DMA */
    16. __HAL_DMA_DISABLE(&hdma_usart1_tx);
    17. if(HAL_UART_Transmit_DMA(&huart1, buf, len) != HAL_OK)
    18. {
    19. Error_Handler();
    20. }
    21. }
    22. /*
    23. *********************************************************************************************************
    24. * 函 数 名: DMA_Usart1_Read
    25. * 功能说明: 串口接收功能函数
    26. * 形 参: Data,len
    27. * 返 回 值: 无
    28. *********************************************************************************************************
    29. */
    30. void DMA_Usart1_Read(uint8_t *Data,uint8_t len)
    31. {
    32. HAL_UART_Receive_DMA(&huart1,Data,len);
    33. }

  • 相关阅读:
    【面试:并发篇28:volatile】有序性
    odoo wizard界面显示带复选框列表及勾选数据获取
    区块链是价值协议
    Java线程池ThreadPoolExecutor极简教程
    为解决bypy大文件上传报错—获取百度云文件直链并使用Aria2上传文件至服务器
    搭建开发环境
    Linux shell编程学习笔记2:我是谁 | who am i ?
    甲骨文严查Java授权,换openJDK要避坑
    硬核剖析ThreadLocal源码,面试官看了直呼内行
    Python练习题:从列表中选取任意个元素求和
  • 原文地址:https://blog.csdn.net/winafa/article/details/133041312