---
title: mcu-stm32-cube-06-配置DMA
date: 2020-05-31 16:39:05
categories:
tags:
- stm32
- cubeMx
- dma
- serial
---

知识

DMA(Direct Memory Access,直接存储器访问) 用于在外设与存储器之间以及存储器与存储器之间提供高速数据传输。

可以在无需任何 CPU 操作的情况下通过 DMA 快速移动数据。这样节省的 CPU 资源可供其它操作使用。

DMA参数

传输模式:数据从哪里搬到哪里。三种可能的传输方向:存储器到外设、外设到存储器或存储器到存储器。

通道选择:数据传输的是走哪条道路。

仲裁器:多个DMA传输时,优先级高的优先传输。

数据长度:每次传输的数据长度,可以1个字节,2个字节(半字),4个字节(字)

指针递增:使能此模式以后,则下次传输的地址将是前一次传输的地址递增 1(对于字节)、2(对于半字)或4(对于字)。

CubeMX 配置 DMA(以 USART2 为例)

配置

添加串口以及中断

Pinout & Configuration页中的Connectivity,选择USART2

  • Mode : Asynchronous(异步); Hardware Flow Control(硬件流控) 选择 Disable

  • Configuration - Parameter Settings中 (任意设置都可以,但通讯双方要匹配)

    • Baud Rate : 波特率,一般使用 115200
    • Word Length : 字长 8
    • Parity: 校验
    • Stop Bits : 停止位
  • Configuration - NVIC Settings中 : 勾选 Enabled (开启中断)

配置DMA

  • Configuration - DMA Settings中,点击ADD分别将USART2_RXUSART_TX添加到DMA Request表中:

    • Priority:设置优先级为Low
    • DMA Request Settings-Mode
      • Normal:只传送一次(选择此项)
      • Circular:不停地传送(循环模式)
      • Data Width:选择Byte

生产中断初始化函数

Pinout & Configuration中,System Core,选择NVIC

  • Configuration - Parameter Settings中 ,确认UsartX global interrupt Enable是勾选的
  • Configuration - Code generation中,确认UsartX global interrupt DMA1 channel 7 global interruptDMA1 channel 6 global interruptSelect for init sequence ordering是勾选的。

收尾工作

填写有关的项目属性、点击右上角的GENERATE CODE

添加代码

stm32f1xx_it.h

添加全局变量

实际上,这些变量可以封装成函数,但是为了减少教程的篇幅,这里没有这么做

/* USER CODE BEGIN EC */
/* 以下变量与DMA接收流程有关 */
volatile uint8_t rx_len = 0; // 获取收到的数据长度
volatile uint8_t recv_end_flag = 0;
uint8_t rx_buffer[200] ;
/* USER CODE END EC */

stm32f1xx_it.c

修改中断处理函数

void USART2_IRQHandler(void)
{
/* USAR CODE BEGIN USART2_IRQn 0 */
uint32_t tmp_flag = 0;
uint32_t temp;
tmp_flag =__HAL_UART_GET_FLAG(&huart2,UART_FLAG_IDLE); //获取IDLE标志位
if((tmp_flag != RESET))//idle标志被置位
{
__HAL_UART_CLEAR_IDLEFLAG(&huart2);//清除标志位
temp = huart2.Instance->SR; //清除状态寄存器SR,读取SR寄存器可以实现清除SR寄存器的功能
temp = huart2.Instance->DR; //读取数据寄存器中的数据
HAL_UART_DMAStop(&huart2); //
temp = hdma_usart2_rx.Instance->CNDTR;// 获取DMA中未传输的数据个数,CNDTR寄存器分析见下面
rx_len = sizeof(rx_buffer) - temp; //总计数减去未传输的数据个数,得到已经接收的数据个数
recv_end_flag = 1; // 接受完成标志位置1
}
/* USAR CODE END USART2_IRQn 0 */
HAL_UART_IRQHandler(&huart2);
/* USAR CODE BEGIN USART2_IRQn 1 */
/* USAR CODE END USART2_IRQn 1 */
}

main.c

引入全局变量

/* USER CODE BEGIN PV */

/* 以下变量与DMA接收流程有关 */
extern volatile uint8_t rx_len; // 获取收到的数据长度
extern volatile uint8_t recv_end_flag;
extern uint8_t rx_buffer[200] ; /* USER CODE END PV */

添加printf重定向

/* USER CODE BEGIN 0 */

#include "stdio.h"

#ifdef __GNUC__
#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
#else
#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
#endif
/* 重定向printf */
PUTCHAR_PROTOTYPE
{
HAL_UART_Transmit(&huart2, (uint8_t*)&ch,1,HAL_MAX_DELAY);
return ch;
}
/* 重定向scanf */
int fgetc(FILE *f)
{
uint8_t ch = 0;
HAL_UART_Receive(&huart2, &ch, 1, 0xffff);
return ch;
} /* USER CODE END 0 */

添加DMA初始化

/* USER CODE BEGIN 2 */

// 在 MX_USART2_UART_Init 之后添加这2行
__HAL_UART_ENABLE_IT(&huart2, UART_IT_IDLE); //使能idle中断
HAL_UART_Receive_DMA(&huart2,rx_buffer, sizeof(rx_buffer)); //打开DMA接收,数据会存入rx_buffer数组中。(删掉会导致第一次DMA异常) /* USER CODE END 2 */

使用(在需要调用的地方,添加以下函数,以main中的while(1)为例):

/* USER CODE BEGIN WHILE */
while (1)
{
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
if(recv_end_flag ==1)
{
//打印接收长度、数据
printf("rx_len=%d\r\n",rx_len);
printf("%s\r\n",rx_buffer);
for(uint8_t i=0;i<rx_len;i++)
{
rx_buffer[i]=0;//清接收缓存
}
rx_len=0;//清除计数
recv_end_flag=0;//清除接收结束标志位
}
HAL_UART_Receive_DMA(&huart2,rx_buffer,sizeof(rx_buffer));//重新打开DMA接收
}
/* USER CODE END 3 */

附录:

typedef struct
{
__IO uint32_t CCR; /*!< DMA stream x configuration register */
__IO uint32_t CNDTR; /*!< DMA stream x **number of data register** */
__IO uint32_t CPAR; /*!< DMA stream x peripheral address register */
__IO uint32_t CMAR; /*!< DMA stream x memory address register */
} DMA_Channel_TypeDef;

STM32 CubeMX 学习:06-配置DMA的更多相关文章

  1. cubemx+stm32串口学习汇总资料

    这篇文章是串口中断的文章--STM32基于CubeMX的高速串口收发程序(中断模式)比较有帮助. http://www.stmcu.org.cn/module/forum/thread-616613- ...

  2. STM32 FSMC学习笔记+补充(LCD的FSMC配置)

    STM32 FSMC学习笔记+补充(LCD的FSMC配置) STM32 FSMC学习笔记 STM32 FSMC的用法--LCD

  3. stm32定时器学习二——PWM设置

    /* STM32 嵌入式学习入门(5)——PWM的实现 上一篇博文介绍了定时器和PWM的基本的原理,本篇博文从代码层面来介绍PWM的具体实现.同样,还是以博主所用的开发板——正点原子开发板STM32F ...

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

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

  5. [FreeRTOS入门] 1.CubeMX中FreeRTOS配置参数及理解

    1.有关优先级 1.1 Configuration --> FreeRTOS MAX_PRIORITIES 设置任务优先级的数量:配置应用程序有效的优先级数目.任何数量的任务都可以共享一个优先级 ...

  6. 基于STM32的学习型通用红外遥控设备的设计实现(三)

    CPU: STM32 调试平台: STM32F103ZET和STM32F103VBT 软件平台: Keil uVision4 电路设计: Altium Designer v6.9 http://blo ...

  7. 【转载-Andrew_qian】stm32中断学习

    [转载]stm32中断学习 中断对于开发嵌入式系统来讲的地位绝对是毋庸置疑的,在C51单片机时代,一共只有5个中断,其中2个外部中断,2个定时/计数器中断和一个串口中断,但是在STM32中,中断数量大 ...

  8. stm32串口学习笔记

    stm32作为现在嵌入式物联网单片机行业中经常要用多的技术,相信大家都有所接触,今天这篇就给大家详细的分析下有关于stm32的出口,还不是很清楚的朋友要注意看看了哦,在最后还会为大家分享有些关于stm ...

  9. STM32 CubeMx使用教程

    一.STM32CubeMX 简介 STM32CubeMX 是 ST 意法半导体近几年来大力推荐的STM32 芯片图形化配置工具,目的就是为了方便开发者, 允许用户使用图形化向导生成C 初始化代码,可以 ...

  10. STM32的SPI口的DMA读写[原创www.cnblogs.com/helesheng]

    SPI是我最常用的接口之一,连接管脚仅为4根:在常见的芯片间通信方式中,速度远优于UART.I2C等其他接口.STM32的SPI口的同步时钟最快可到PCLK的二分之一,单个字节或字的通信时间都在us以 ...

随机推荐

  1. vue+vant+js实现购物车原理小demo(基础版)

    电商毕业设计里的一个购物车demo,拿vue+vant需要写的核心计算代码只有12行.效果图: main.js: Vue.use(Stepper); .vue文件 <template> & ...

  2. Intel Pentium III CPU(Coppermine, Tualatin) L2 Cache Latency, Hardware Prefetch特性调查

    这几天,偶然的机会想到了困扰自己和其他网友多年的Intel Pentium III系列处理器缓存延迟(L2 Cache Latency),以及图拉丁核心版本是否支持硬件预取(Hardware Pref ...

  3. CentOS-7卸载了python2.7,yum不可用的解决方法

    1.mount挂载iso镜像 [root@localhost software]# mount -t iso9660 -o loop CentOS-7-x86_64-DVD-2003.iso /med ...

  4. 【Python Web】flask视频流

    这篇文档,完全借鉴miguelgrinberg的博客. https://blog.miguelgrinberg.com/post/flask-video-streaming-revisited 想看具 ...

  5. go1.18泛型全部教程

    目录 go1.18泛型全部教程 一 什么是泛型 二 Golang中的泛型 三 泛型语法详解 3.1 泛型的语法 3.2 Constraint(约束)是什么 3.3 自定义constraint(约束) ...

  6. 详解Python 中可视化数据分析工作流程

    本文分享自华为云社区<Python 可视化数据分析从数据获取到洞见发现的全面指南>,作者:柠檬味拥抱. 在数据科学和分析的领域中,可视化是一种强大的工具,能够帮助我们理解数据.发现模式,并 ...

  7. 80x86汇编—80x86架构

    文章目录 计算机如何工作 存储器 逻辑地址到物理地址 寄存器 数据寄存器使用细节 其他知识点细节 堆栈Stack 标志寄存器 中断 汇编入门简单,深入难 使用8086架构进行学习,本章节如果没有学过计 ...

  8. Haproxy+Nginx+Tomcat实现动静分离页面

    一.Haproxy概述: 二.Haproxy原理实现: 三.Nginx.LVS.Haproxy对比: 四.Haproxy配置文件讲解: 五.案例:Haproxy+Nginx+Tomcat搭建高可用集群 ...

  9. 实战-mongodb副本集搭建以及整合springboot使用

    一 mongodb介绍 MongoDB是一个基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案.  Nosql 技术门类 redis 内存型 mongod ...

  10. CSS 溢出文本

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...