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

STM32-笔记35-DMA(直接存储器访问)

一、什么叫DMA?

        DMA(Direct Memory Access,直接存储器访问)提供在外设与内存、存储器和存储器之间的高速数据传输使用。它允许不同速度的硬件装置来沟通,而不需要依赖于CPU,在这个时间中,CPU对于内存的工作来说就无法使用。

简单来说,就是一个数据的搬运工

而搬运的路径有三种方式:如下所示

  • 存储器→存储器(例如:复制某特别大的数据buf)
  • 存储器→外设 (例如:将某数据buf写入串口TDR寄存器)
  • 外设→存储器 (例如:将串口RDR寄存器写入某数据buf)

        存储器我们知道,就是内存,存储器到存储器之间的搬运,也就是从内存中一地方搬运到内存中另一个地方。

        那么这里的外设是指什么呢?

        外设指的是spi、usart、iic、adc 等基于APB1 、APB2或AHB时钟的外设,而这里的存储器包括 自身的闪存(flash)或者内存(SRAM)以及外设的存储设备都可以作为访问地源或者目的

具体在参考手册中DMA框图中可以看见外设有哪些

 二、DMA存在的意义?

        代替 CPU 搬运数据,为 CPU 减负。

  • 1. 数据搬运的工作比较耗时间;
  • 2. 数据搬运工作时效要求高(有数据来就要搬走);
  • 3. 没啥技术含量(CPU 节约出来的时间可以处理更重要的事)。

三、搬运的流程

存储器→存储器

存储器→外设

外设→存储器

四、DMA控制器-通道

STM32F103 有 2 个 DMA 控制器,DMA1 有 7 个通道,DMA 2 有 5 个通道。
一个通道每次只能搬运一个外设的数据!! 如果同时有多个外设的 DMA 请求,则按照优先级进行响应。
STM32F103C8T6 只有 DMA1 !
DMA1有7个通道:

每个通道传输特定外设的数据

五、DMA优先级管理

优先级管理采用软件+硬件:
        软件: 每个通道的优先级可以在DMA_CCRx寄存器中设置,有4个等级
最高级>高级>中级>低级
        硬件: 如果2个请求,它们的软件优先级相同,则较低编号的通道比较高编号的通道有较高的优先权。
        比如:如果软件优先级相同,通道2优先于通道4

六、DMA传输方式

 

七、DMA寄存器及库函数介绍
 

八、小实验1:DMA内存到内存数据搬运

实验目的

使用DMA将一个大数组的数据搬运到另一个位置。

复制项目文件19-串口打印功能

重命名为40-DMA实验(内存到内存)

新建文件夹dma dma.c  dma.h

打开项目文件

加载文件

main.c

#include "sys.h"
#include "delay.h"
#include "led.h"
#include "uart1.h"
#include "dma.h"

int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
    led_init();//初始化led灯
    uart1_init(115200);
    dma_init();
    printf("hello word!\r\n");
    dma_transmit();
    
//    led1_ON();
//    led1_OFF();
    while(1)
    { 
        led1_ON();
        led2_OFF();
        delay_ms(500);
        led1_OFF();
        led2_ON();
        delay_ms(500);
    }
}

dma.c

#include "dma.h"
#include "stdio.h"
#define BUF_SIZE 16

uint32_t src_buf[BUF_SIZE] = {
    0x00000000,0x11111111,0x22222222,0x33333333,
    0x44444444,0x55555555,0x66666666,0x77777777,
    0x88888888,0x99999999,0xAAAAAAAA,0xBBBBBBBB,
    0xCCCCCCCC,0xDDDDDDDD,0xEEEEEEEE,0xFFFFFFFF
};
uint32_t dst_buf[BUF_SIZE] = {0};

DMA_HandleTypeDef dma_handle = {0};
//初始化dma函数
void dma_init(void)
{
    __HAL_RCC_DMA1_CLK_ENABLE();//使能DMA1
    dma_handle.Instance = DMA1_Channel1;//选择DMA1通道1(由于是内存传递内存所以不管哪一个通道都可以)
    dma_handle.Init.Direction = DMA_MEMORY_TO_MEMORY;//方向:内存到内存
    //内存的设置
    dma_handle.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;//内存数据对其方式:字节方式对齐
    dma_handle.Init.MemInc = DMA_MINC_ENABLE;//内存增量启动
    //外设的设置
    dma_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;//外设数据对其方式
    dma_handle.Init.PeriphInc = DMA_PINC_ENABLE;//外设增量模式启动
    
    dma_handle.Init.Mode = DMA_NORMAL;//DMA的模式:正常模式和循环模式
    dma_handle.Init.Priority = DMA_PRIORITY_MEDIUM;//DMA优先级:中级(只有一个DMA设置成什么无所谓)
    
    HAL_DMA_Init(&dma_handle);
}
//准备数据传输
void dma_transmit(void)
{
    //启动DMA传输
    HAL_DMA_Start(&dma_handle,(uint32_t)src_buf,(uint32_t)dst_buf,sizeof(uint32_t) * BUF_SIZE);
    //查询DMA传输状态
    while(__HAL_DMA_GET_FLAG(&dma_handle,DMA_FLAG_TC1) == RESET);
    //打印数据
    int i;
    for(i=0;i<BUF_SIZE;i++)
    {
        printf("buf[%d]:%X\r\n",i,dst_buf[i]);
    }
    
}

dma.h

#ifndef __DMA_H__
#define __DMA_H__

#include "sys.h"

void dma_init(void);
void dma_transmit(void);
    
#endif

九、小实验2:DMA内存到外设数据搬运

实验目的

使用DMA将一个大数组的数据通过串口1发送 。

内存通过DMA将数据搬运到外设中

USART1-TX负责将内存中的数据按照指定的格式和波特率发送到外部设备‌

复制项目文件40-DMA实验(内存到内存)

重命名项目文件41-DMA实验(内存到外设)

打开项目文件

main.c

#include "sys.h"
#include "delay.h"
#include "led.h"
#include "uart1.h"
#include "dma.h"

uint8_t send_buf[1000] = {0};
extern UART_HandleTypeDef uart1_handle;

int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
    led_init();//初始化led灯
    uart1_init(115200);
    dma_init();
    //printf("hello word!\r\n");
    
    int i = 0;
    for(i=0;i<1000;i++)
    {
        send_buf[i] = 'A';
    }
    //串口句柄,发送的数据,发送的个数
    HAL_UART_Transmit_DMA(&uart1_handle,send_buf,1000);//以DMA的方式发送数据
    while(1)
    { 
        led1_ON();
        led2_OFF();
        delay_ms(500);
        led1_OFF();
        led2_ON();
        delay_ms(500);
    }
}

dma.c

#include "dma.h"
#include "stdio.h"

extern UART_HandleTypeDef uart1_handle;
DMA_HandleTypeDef dma_handle = {0};
//初始化dma函数
void dma_init(void)
{
    __HAL_RCC_DMA1_CLK_ENABLE();//使能DMA1
    dma_handle.Instance = DMA1_Channel4;//选择DMA1通道4,内存到外设通道4-以DMA的方式发送(tx)给串口1数据
    dma_handle.Init.Direction = DMA_MEMORY_TO_PERIPH;//方向:内存到外设
    //内存的设置
    dma_handle.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;//内存数据对其方式:字节方式对齐
    dma_handle.Init.MemInc = DMA_MINC_ENABLE;//内存增量启动:地址递增
    //外设的设置
    dma_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;//外设数据对其方式
    dma_handle.Init.PeriphInc = DMA_PINC_DISABLE;//外设不递增
    
    dma_handle.Init.Mode = DMA_NORMAL;//DMA的模式:正常模式和循环模式
    dma_handle.Init.Priority = DMA_PRIORITY_MEDIUM;//DMA优先级:中级(只有一个DMA设置成什么无所谓)
    
    HAL_DMA_Init(&dma_handle);
    //链接:内存链接到串口1:外设的句柄,成员变量,dma的句柄
    __HAL_LINKDMA(&uart1_handle,hdmatx,dma_handle);
}

dma.h

#ifndef __DMA_H__
#define __DMA_H__

#include "sys.h"

void dma_init(void);
    
#endif

结果:

十、小实验3:DMA外设到内存数据搬运

实验目的

使用DMA接收串口1发的数据。

串口通过DMA将数据搬运到内存中

USART1-RX负责将外部设备‌中的数据按照指定的格式和波特率发送到内存,

也就是内存接收外部设备中的数据

所以这里的USART1-RX和USART1-TX是针对内存来讲(暂时先这么记)

复制项目文件41-DMA实验(内存到外设)

重命名项目文件42-DMA实验(外设到内存)

main.c

#include "sys.h"
#include "delay.h"
#include "led.h"
#include "uart1.h"
#include "dma.h"

int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
    led_init();//初始化led灯
    uart1_init(115200);
    dma_init();
    printf("hello word!\r\n");
    
    while(1)
    { 
        led1_ON();
        led2_OFF();
        delay_ms(500);
        led1_OFF();
        led2_ON();
        delay_ms(500);
    }
}

dma.c

#include "dma.h"
#include "stdio.h"
#include "uart1.h"

extern UART_HandleTypeDef uart1_handle;
DMA_HandleTypeDef dma_handle = {0};
extern uint8_t uart1_rx_buf[UART1_RX_BUF_SIZE];                                    /* UART1接收缓冲区 */
                                               

//初始化dma函数
void dma_init(void)
{
    __HAL_RCC_DMA1_CLK_ENABLE();//使能DMA1
    dma_handle.Instance = DMA1_Channel5;//选择DMA1通道5,外设到内存通道5-以DMA的方式发送(rx)给串口1数据
    dma_handle.Init.Direction = DMA_PERIPH_TO_MEMORY;//方向:外设到内存
    //内存的设置
    dma_handle.Init.MemDataAlignment = DMA_MDATAALIGN_BYTE;//内存数据对其方式:字节方式对齐
    dma_handle.Init.MemInc = DMA_MINC_ENABLE;//内存增量启动:地址递增
    //外设的设置
    dma_handle.Init.PeriphDataAlignment = DMA_PDATAALIGN_BYTE;//外设数据对其方式
    dma_handle.Init.PeriphInc = DMA_PINC_DISABLE;//外设不递增
    
    dma_handle.Init.Mode = DMA_NORMAL;//DMA的模式:正常模式和循环模式
    dma_handle.Init.Priority = DMA_PRIORITY_MEDIUM;//DMA优先级:中级(只有一个DMA设置成什么无所谓)
    
    HAL_DMA_Init(&dma_handle);
    //链接:内存链接到串口1:外设的句柄,成员变量,dma的句柄
    __HAL_LINKDMA(&uart1_handle,hdmarx,dma_handle);
    //串口通过DMA接收
    HAL_UART_Receive_DMA(&uart1_handle,uart1_rx_buf,UART1_RX_BUF_SIZE);//将数据接收到uart1_rx_buf缓冲区里面长度是UART1_RX_BUF_SIZE这么长
}

dma.h

#ifndef __DMA_H__
#define __DMA_H__

#include "sys.h"

void dma_init(void);
    
#endif

结果:


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

相关文章:

  • 腾讯云AI代码助手编程挑战赛-图片转换工具
  • Improving Language Understanding by Generative Pre-Training GPT-1详细讲解
  • 腾讯云AI代码助手编程挑战赛-凯撒密码解码编码器
  • vulnhub靶场【DC系列】之7
  • 【和春笋一起学C++】文本输入与读取(二)
  • 应急响应——Windows / Linux 排查笔记
  • Approaching a Chatbot Service - Part 3: Bot Model
  • flask后端开发(13):登录功能后端实现和钩子函数
  • 扩散模型论文概述(三):Stability AI系列工作【学习笔记】
  • 基于RK3568/RK3588大车360度环视影像主动安全行车辅助系统解决方案,支持ADAS/DMS
  • 关于常见的负载均衡方法都有什么?
  • labelme转yolo格式
  • 计算机网络 (23)IP层转发分组的过程
  • R语言的循环实现
  • 前端开发 vue 中如何实现 u-form 多个form表单同时校验
  • Estimator (Statistic for Machine Learning)
  • 深入探讨 Android 中的 AlarmManager:定时任务调度及优化实践
  • AI赋能房地产:利用AI前端技术提升VR/AR浏览体验
  • IO模型与NIO基础
  • 【Uniapp-Vue3】v-if条件渲染及v-show的选择对比
  • 第07章 存储管理(一)
  • 第82期 | GPTSecurity周报
  • 基于Vite+TS初始项目 | 不断更新
  • vue 项目集成 electron 和 electron 打包及环境配置
  • 基于R语言的DICE模型
  • gitee 使用教程