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

STM32教程:DMA运用及代码(基于STM32F103C8T6最小系统板标准库开发)*详细教程*

前言:

本文章主要介绍了STM32微控制器的DMA外设的原理运用、库函数介绍、代码编写实现DMA转运的功能。

DMA介绍:

STM32 的 DMA(Direct Memory Access,直接内存访问)是一种重要机制,能让数据在不经过 CPU 的情况下,直接在内存和外设间传输。

传统数据传输时,CPU 要负责搬运数据,会占用大量时间和资源。而 DMA 工作时,它会接管总线控制权,外设与内存间的数据传输由 DMA 控制器完成。比如,当 ADC 转换完成数据后,DMA 可直接将数据从 ADC 的数据寄存器传输到内存指定区域。

这极大减轻了 CPU 负担,使 CPU 可在 DMA 传输数据期间处理其他任务,提升了系统的整体性能和效率,尤其适用于大数据量、高速率的数据传输场景。

根据DMA基本结构来编写配置DMA的代码


我们以程序中,以数组的转运的来作为DMA转运的例子

大体流程:

1、RCC开启时钟

2、调用DMA_Init,初始化

3、开关控制

4、如果选择硬件触发,要在对应的外设调用XXX_DMACmd函数

5、如果需要DMA的中断,就调用DMA_ITConfig函数


DMA库函数介绍

在库函数dma.h

恢复初始化配置

void DMA_DeInit(DMA_Channel_TypeDef* DMAy_Channelx);

初始化

void DMA_Init(DMA_Channel_TypeDef* DMAy_Channelx, DMA_InitTypeDef* DMA_InitStruct);

结构体初始化

void DMA_StructInit(DMA_InitTypeDef* DMA_InitStruct);

使能DMA

void DMA_Cmd(DMA_Channel_TypeDef* DMAy_Channelx, FunctionalState NewState);

中断输出使能

void DMA_ITConfig(DMA_Channel_TypeDef* DMAy_Channelx, uint32_t DMA_IT, FunctionalState NewState);

DMA设置当前数据寄存器

这个函数,就是给这个传输寄存器写数据的

void DMA_SetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx, uint16_t DataNumber); 

DMA获取当前数据寄存器

返回传输计数器的值

uint16_t DMA_GetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx);

最后四个函数:

获取标志位状态、清除标志位、获取中断状态、清除中断挂起位

FlagStatus DMA_GetFlagStatus(uint32_t DMAy_FLAG);
void DMA_ClearFlag(uint32_t DMAy_FLAG);
ITStatus DMA_GetITStatus(uint32_t DMAy_IT);
void DMA_ClearITPendingBit(uint32_t DMAy_IT);


详细步骤

创建MyDMA.c文件,编写程序

1、开启时钟

因为DMA是AHB总线的设备,所以要用AHB开启时钟函数

	/*开启DMA时钟*/RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);  

2、调用DMA_Init,初始化

包括外设和存储器站点的起始地址、数据宽度、地址是否自增、方向、传输计数器、是否需要自动重装、选择触发源、通道优先级(所有的参数,通过一个机构体,就可以配置好了)

	/*初始化DMA*/DMA_InitTypeDef DMA_InitStructure;DMA_InitStructure.DMA_PeripheralBaseAddr = AddrA;  //外设站点的起始地址DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //外设站点的数据宽度DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;	//外设站点的是否自增DMA_InitStructure.DMA_MemoryBaseAddr = AddrB;   //存储器站点的起始地址DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;	//存储器站点数据宽度DMA_InitStructure.DMA_MemoryInc	= DMA_MemoryInc_Enable;		//存储器站点是否自增DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;     // 传输方向DMA_InitStructure.DMA_BufferSize = Size;	//缓存器大小(传输计数器)DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;	//传输模式(是否使用自动重装)DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;	 //硬件触发还是软件触发DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;  //优先级DMA_Init(DMA1_Channel1,&DMA_InitStructure);

注意在头文件声明函数

#ifndef __MYDMA_H
#define __MYDMA_Hvoid MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t Size);#endif

编写主函数文件,调用实验一下

实现DataA数组到DataB数组的转运

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MyDMA.h"uint8_t DataA[] = {0x01,0x02,0x03,0x04};
uint8_t DataB[] = {0,0,0,0};int main(void)
{/*模块初始化*/OLED_Init();		//OLED初始化/*OLED显示*///转运前的数据OLED_ShowHexNum(1,1,DataA[0],2);OLED_ShowHexNum(1,4,DataA[1],2);OLED_ShowHexNum(1,7,DataA[2],2);OLED_ShowHexNum(1,10,DataA[3],2);OLED_ShowHexNum(2,1,DataB[0],2);OLED_ShowHexNum(2,4,DataB[1],2);OLED_ShowHexNum(2,7,DataB[2],2);OLED_ShowHexNum(2,10,DataB[3],2);MyDMA_Init((uint32_t)DataA,(uint32_t)DataB,4);  //转运数据//转运后的数据OLED_ShowHexNum(3,1,DataA[0],2);OLED_ShowHexNum(3,4,DataA[1],2);OLED_ShowHexNum(3,7,DataA[2],2);OLED_ShowHexNum(3,10,DataA[3],2);OLED_ShowHexNum(4,1,DataB[0],2);OLED_ShowHexNum(4,4,DataB[1],2);OLED_ShowHexNum(4,7,DataB[2],2);OLED_ShowHexNum(4,10,DataB[3],2);while (1){}
}

实验现象:

可以看到:一二行是转运前的数据,三四行是转运后的数据

再封装一个DMA转运数据的函数,调用更加方便


void MyDMA_Transfer(void)
{DMA_Cmd(DMA1_Channel1,DISABLE);   //失能DMA_SetCurrDataCounter(DMA1_Channel1,MyDMA_Size);  //给传输寄存器赋值DMA_Cmd(DMA1_Channel1,ENABLE);    //使能/*等待转运完成*/while(DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);/*清除标志位*/DMA_ClearFlag(DMA1_FLAG_TC1);
}


附录(源代码):

MyDMA.c

#include "stm32f10x.h"                  // Device headeruint16_t MyDMA_Size;void MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t Size)
{MyDMA_Size = Size;/*开启DMA时钟*/RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);  /*初始化DMA*/DMA_InitTypeDef DMA_InitStructure;DMA_InitStructure.DMA_PeripheralBaseAddr = AddrA;  //外设站点的起始地址DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //外设站点的数据宽度DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;	//外设站点的是否自增DMA_InitStructure.DMA_MemoryBaseAddr = AddrB;   //存储器站点的起始地址DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;	//存储器站点数据宽度DMA_InitStructure.DMA_MemoryInc	= DMA_MemoryInc_Enable;		//存储器站点是否自增DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;     // 传输方向DMA_InitStructure.DMA_BufferSize = Size;	//缓存器大小(传输计数器)DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;	//传输模式(是否使用自动重装)DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;	 //硬件触发还是软件触发DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;  //优先级DMA_Init(DMA1_Channel1,&DMA_InitStructure);/*失能DMA*/DMA_Cmd(DMA1_Channel1,DISABLE);}void MyDMA_Transfer(void)
{DMA_Cmd(DMA1_Channel1,DISABLE);   //失能DMA_SetCurrDataCounter(DMA1_Channel1,MyDMA_Size);  //给传输寄存器赋值DMA_Cmd(DMA1_Channel1,ENABLE);    //使能/*等待转运完成*/while(DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);/*清除标志位*/DMA_ClearFlag(DMA1_FLAG_TC1);
}

MyDMA.h

#ifndef __MYDMA_H
#define __MYDMA_Hvoid MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t Size);
void MyDMA_Transfer(void);#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MyDMA.h"uint8_t DataA[] = {0x01,0x02,0x03,0x04};
uint8_t DataB[] = {0,0,0,0};int main(void)
{/*模块初始化*/OLED_Init();		//OLED初始化MyDMA_Init((uint32_t)DataA,(uint32_t)DataB,4);  //DMA初始化/*OLED显示*/OLED_ShowString(1,1,"DataA");OLED_ShowString(3,1,"DataA");	OLED_ShowHexNum(1,8,(uint32_t)DataA,8);OLED_ShowHexNum(3,8,(uint32_t)DataB,8);//转运前的数据OLED_ShowHexNum(2,1,DataA[0],2);OLED_ShowHexNum(2,4,DataA[1],2);OLED_ShowHexNum(2,7,DataA[2],2);OLED_ShowHexNum(2,10,DataA[3],2);OLED_ShowHexNum(4,1,DataB[0],2);OLED_ShowHexNum(4,4,DataB[1],2);OLED_ShowHexNum(4,7,DataB[2],2);OLED_ShowHexNum(4,10,DataB[3],2);while (1){DataA[0] ++;DataA[1] ++;	DataA[2] ++;DataA[3] ++;//转运前的数据OLED_ShowHexNum(2,1,DataA[0],2);OLED_ShowHexNum(2,4,DataA[1],2);OLED_ShowHexNum(2,7,DataA[2],2);OLED_ShowHexNum(2,10,DataA[3],2);OLED_ShowHexNum(4,1,DataB[0],2);OLED_ShowHexNum(4,4,DataB[1],2);OLED_ShowHexNum(4,7,DataB[2],2);OLED_ShowHexNum(4,10,DataB[3],2);Delay_ms(1000);MyDMA_Transfer();//转运后的数据OLED_ShowHexNum(2,1,DataA[0],2);OLED_ShowHexNum(2,4,DataA[1],2);OLED_ShowHexNum(2,7,DataA[2],2);OLED_ShowHexNum(2,10,DataA[3],2);OLED_ShowHexNum(4,1,DataB[0],2);OLED_ShowHexNum(4,4,DataB[1],2);OLED_ShowHexNum(4,7,DataB[2],2);OLED_ShowHexNum(4,10,DataB[3],2);Delay_ms(1000);}
}

相关文章:

  • 杨校老师竞赛课之C++备战蓝桥杯初级组省赛
  • 从零开始开发纯血鸿蒙应用之NAPI
  • day16 numpy和shap深入理解
  • Linux 文件系统深度解析
  • 设计模式(结构型)-组合模式
  • 深入探索 51 单片机:从入门到实践的全面指南
  • 《MATLAB实战训练营:从入门到工业级应用》高阶挑战篇-《用无人机仿真玩转PID控制:MATLAB四旋翼仿真建模全攻略》
  • Monster Hunter Rise 怪物猎人 崛起 [DLC 解锁] [Steam] [Windows SteamOS]
  • 数据类型:String
  • Linux:权限的理解
  • 湖北理元理律师事务所:债务法律服务的民生价值重构
  • 人工智能应用:从技术突破到生态重构的演进之路
  • MySQL JOIN详解:掌握数据关联的核心技能
  • 使用 vllm 部署 Llama3-8b-Instruct
  • simulink使能子系统的四种配置
  • 【云备份】服务端业务处理模块设计与实现
  • 2.在Openharmony写hello world
  • 解决跨域的4种方法
  • Redis 中简单动态字符串(SDS)的深入解析
  • 基于Redis实现优惠券秒杀——第3期(分布式锁-Redisson)
  • 特朗普:对所有在国外制作进入美国的电影征收100%关税
  • 浙江一文旅局长五一亲自带团,去年专门考取了导游证
  • 人民日报头版:让青春之花绽放在祖国和人民最需要的地方
  • 英国传统两党受挫地方选举后反思,改革党异军突起“突破想象”
  • 2024年境内酒店住宿行业指标同比下滑:酒店行业传统增长模式面临挑战
  • 陈颖已任上海黄浦区委常委、统战部部长