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

嵌入式学习(18)-stm32F407串口接收空闲中断+DMA

一、概述

在一些一次性接收大批量数据的引用场合,如果使用接收中断会频繁的进入接收中断影响代码的运行效率。为了解决这个问题可以使用串口的空闲中断+DMA实现。

二、应用

在网上招了一些例程在STM32F407的平台上都没有跑通会出现各种异常,主要原因还是库的版本有更新可能和当前的工程不匹配,经过几天的煎熬终于调通了流程。

三、代码实现

1、初始化程序

void USART3_Init(uint32_t baudrate)
{/* IO 及 时钟配置 */USART3_TX_GPIO_CLK_ENABLE(); /* 使能 串口TX脚 时钟 */USART3_RX_GPIO_CLK_ENABLE(); /* 使能 串口RX脚 时钟 */USART3_CLK_ENABLE();      /* 使能 串口 时钟 */GPIO_InitTypeDef gpio_init_struct;gpio_init_struct.Pin = USART3_TX_GPIO_PIN;gpio_init_struct.Mode = GPIO_MODE_AF_PP;gpio_init_struct.Pull = GPIO_PULLUP;gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH;gpio_init_struct.Alternate = USART3_TX_GPIO_AF;              /* 复用为串口 */HAL_GPIO_Init(USART3_TX_GPIO_PORT, &gpio_init_struct); /* 串口TX 脚 模式设置 *///gpio_init_struct.Pin = USART3_RX_GPIO_PIN;gpio_init_struct.Alternate = USART3_RX_GPIO_AF;              /* 复用为USART3 */HAL_GPIO_Init(USART3_RX_GPIO_PORT, &gpio_init_struct); /* 串口RX 脚  *//* USART 初始化设置 */USART3_handler.Instance = USART3;                  /* 选择串口屏对应的串口 */USART3_handler.Init.BaudRate = baudrate;               /* 波特率 */USART3_handler.Init.WordLength = UART_WORDLENGTH_8B;   /* 字长为8位数据格式 */USART3_handler.Init.StopBits = UART_STOPBITS_1;        /* 一个停止位 */USART3_handler.Init.Parity = UART_PARITY_NONE;         /* 无奇偶校验位 */USART3_handler.Init.HwFlowCtl = UART_HWCONTROL_NONE;   /* 无硬件流控 */USART3_handler.Init.Mode = UART_MODE_TX_RX;            /* 收发模式 */HAL_UART_Init( &USART3_handler);                       /* 使能对应的串口, 但会调用MSp */
//    __HAL_UART_DISABLE_IT( &USART3_handler, UART_IT_TC);__HAL_UART_ENABLE_IT( &USART3_handler, UART_IT_RXNE);  /* 开启接收中断 */__HAL_UART_ENABLE_IT(&USART3_handler, UART_IT_IDLE);    // 使能串口接收空闲中断HAL_NVIC_SetPriority(USART3_IRQn, 3, 3);              /* 抢占优先级3,子优先级3 */HAL_NVIC_EnableIRQ(USART3_IRQn);                      /* 使能USART1中断 */	__HAL_RCC_DMA1_CLK_ENABLE();// 低优先级
//    __HAL_LINKDMA(&USART3_handler, hdmatx, g_dma_handle);   /* 将DMA与USART3联系起来(发送DMA) */	 
__HAL_LINKDMA(&USART3_handler, hdmarx, g_dma_handle); g_dma_handle.Instance = DMA1_Stream1;                    /* 数据流选择 */g_dma_handle.Init.Channel = DMA_CHANNEL_4;                               /* DMA通道选择 */g_dma_handle.Init.Direction = DMA_PERIPH_TO_MEMORY;           /* 外设到存储器 */g_dma_handle.Init.PeriphInc = DMA_PINC_DISABLE;               /* 外设非增量模式 */g_dma_handle.Init.MemInc = DMA_MINC_ENABLE;                   /* 存储器增量模式 */g_dma_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;  /* 外设数据长度:8位 */g_dma_handle.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;     /* 存储器数据长度:8位 */g_dma_handle.Init.Mode = DMA_NORMAL;                          /* 外设流控模式 */g_dma_handle.Init.Priority = DMA_PRIORITY_MEDIUM;             /* 中等优先级 */g_dma_handle.Init.FIFOMode = DMA_FIFOMODE_DISABLE;            /* 关闭FIFO模式 */g_dma_handle.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;    /* FIFO阈值配置 */g_dma_handle.Init.MemBurst = DMA_MBURST_SINGLE;               /* 存储器突发单次传输 */g_dma_handle.Init.PeriphBurst = DMA_PBURST_SINGLE;            /* 外设突发单次传输 */	
//	  HAL_DMA_Init(&g_dma_handle);                          // 将DMA与USART3联系起来(接收DMA)HAL_UART_Receive_DMA(&USART3_handler, g_rcvDataBuf, MAX_BUF_SIZE);             // 开启DMA接收HAL_DMA_Start(&g_dma_handle,(uint32_t)&USART3->DR,(uint32_t)&g_rcvDataBuf,MAX_BUF_SIZE);}

2、中断处理

void USART3_IRQHandler(void)
{
//     printf("t");uint8_t res;uint16_t PACKET_DATA_LEN;uint16_t t;__HAL_UNLOCK( &USART3_handler);//    if ((__HAL_UART_GET_FLAG( &USART3_handler, UART_FLAG_RXNE) != RESET)) /* 接收到数据 */
//    {
//        HAL_UART_Receive( &USART3_handler, &res, 1, 1000);Buffer_Push( &RF_Buffer, res); //接收数据,
//        		printf("%c",res);
//    }	if (__HAL_UART_GET_FLAG(&USART3_handler, UART_FLAG_IDLE) != RESET)                 // 获取接收IDLE标志位是否被置位{   printf("进入空闲中断 !\r\n");          __HAL_UART_CLEAR_IDLEFLAG(&USART3_handler); 
//		HAL_UART_DMAStop(&USART3_handler); 	  /* 异常 */	HAL_DMA_Abort(&g_dma_handle);PACKET_DATA_LEN = (MAX_BUF_SIZE - __HAL_DMA_GET_COUNTER(&g_dma_handle));t=__HAL_DMA_GET_COUNTER(&g_dma_handle);	printf("使用存储空间:%d 剩余存储空间:%d\r\n",PACKET_DATA_LEN,t);printf("接收到的数据:%s",g_rcvDataBuf);HAL_UART_Receive_DMA(&USART3_handler, (uint8_t *)g_rcvDataBuf, MAX_BUF_SIZE);  HAL_DMA_Start(&g_dma_handle,(uint32_t)&USART3->DR,(uint32_t)&g_rcvDataBuf,MAX_BUF_SIZE);// 重新开启DMA传输memset(g_rcvDataBuf, 0, sizeof(g_rcvDataBuf));}}

3、测试结果

4、源码链接

STM32F407基于串口空闲中断和DMA的实现可以实现大批量数据的接收资源-CSDN文库

http://www.lryc.cn/news/503860.html

相关文章:

  • b站视频爬虫-词云分析
  • 如何防止订单二次重复支付?
  • LeetCode 24反转链表
  • 用python的flask写的一个MQTT中转功能,http的方式发送数据和接收数据
  • img引入svg如何修改颜色
  • 计算机毕业设计PySpark+PyFlink+Hive地震预测系统 地震数据分析可视化 地震爬虫 大数据毕业设计 Hadoop 机器学习 深度学习
  • 【Python】使用Numpy实现余弦相似度计算
  • nginx中的root和alias的区别
  • 探索Telnet:实现Windows远程登录Ubuntu的实践指南
  • 在 Vue 2 中隐藏页面元素的方法
  • 【Java】Java8的4个函数式接口简单教程
  • 计算机组成原理与系统结构——微程序控制
  • 【Swift】集合类型 - 数组、集合、字典
  • 3D 视觉定位技术:汽车零部件制造的智能变革引擎
  • 操作系统的基本认识
  • 使用pycharm连接远程服务器
  • 【Linux SH脚本】LinuxCheck 应急检查信息脚本
  • apifox创建一个mock接口
  • 设计一个基础JWT的多开发语言分布式电商系统
  • 委托(Delegate)与事件(Event)-(上篇)
  • Scala根据身份证前两位数判断地区
  • freeswitch(开启支持视频H264通话)
  • 启发式搜索算法和优化算法的区别
  • 数据结构初阶---二叉树---堆
  • 微信小程序中 crypto-js 加解密全攻略
  • 单片机的软件开发环境
  • 【echarts】数据过多时可以左右滑动查看(可鼠标可滚动条)
  • Python 实现对人的行为预测
  • 使用枚举实现单例模式,不会反序列化破坏攻击,不会被反射破坏攻击。(附带枚举单例的简单实现)
  • scala隐式转换