STM32实验DMA数据搬运小助手

这篇具有很好参考价值的文章主要介绍了STM32实验DMA数据搬运小助手。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

本次实验做的是将一个数组的内容利用DMA数据搬运小助手搬运到另外一个数组中去。

STM32实验DMA数据搬运小助手,STM32,stm32,嵌入式硬件,单片机

最后的实验结果:

STM32实验DMA数据搬运小助手,STM32,stm32,嵌入式硬件,单片机

可以看到第四行的数据就都不是0了,成功搬运了过来。

DMA实现搬运的步骤其实不是很复杂,复杂的是结构体参数:

STM32实验DMA数据搬运小助手,STM32,stm32,嵌入式硬件,单片机

整个步骤为:

第一步:RCC开启DMA的时钟

第二步:调用DMA_Init,初始化各个参数 (包括外设和存储器的起始地址,数据宽度,地址是否自增,方向,传输计数器,是否需要自动重装,选择触发源,通道优先级)

第三步:调用DMA_CMD,通道使能(要在对应的外设调用XXX_DMACmd开启一下触发信号的输出,如果需要DMA的中断,就调用DMA_ITConfig,开启中断输出,再在NVIC里,配置中断通道,最后写中断函数就行了)

下面介绍一下DMA相关的库函数:

void DMA_DeInit(DMA_Channel_TypeDef* DMAy_Channelx);

//恢复缺省配置

void DMA_Init(DMA_Channel_TypeDef* DMAy_Channelx, DMA_InitTypeDef* DMA_InitStruct); //DMA初始化

void DMA_StructInit(DMA_InitTypeDef* DMA_InitStruct);

//DMA结构体初始化

void DMA_Cmd(DMA_Channel_TypeDef* DMAy_Channelx, FunctionalState NewState);

//DMA使能/失能

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

//中断输出使能/失能

void DMA_SetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx, uint16_t DataNumber); //DMA设置当前数据寄存器

uint16_t DMA_GetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx);

//DMA获取当前数据寄存器的值

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的文件:

#include "stm32f10x.h"                  // Device header


void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{
	//第一步:RCC开启DMA的时钟
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	
	//第二步:调用DMA_Init,初始化各个参数(包括外设和存储器的起始地址,数据宽度,地址是否自增,方向,传输计数器,是否需要自动重装,选择触发源,通道优先级)
	DMA_InitTypeDef DMA_InitStruct;
	DMA_InitStruct.DMA_PeripheralBaseAddr = AddrA;
	DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;   //数据宽度,按字节的宽度搬运
	DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Enable;    //启用地址++自增
	// 以上是外设站点(数据来源)的起始地址、数据宽度、是否自增。
	
	DMA_InitStruct.DMA_MemoryBaseAddr = AddrB;
	DMA_InitStruct.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;  //数据宽度,按字节的宽度粘贴
	DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;       //启用地址++自增
	//以上3条是存储器(目的地)的起始地址、数据宽度、是否自增。
	
	DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;    //传输方向
	DMA_InitStruct.DMA_BufferSize = Size;  //缓存区大小,其实就是传输计数器   取用传进来的参数
	DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;   //传输模式,其实就是是否启用自动重装     不自动重装
	DMA_InitStruct.DMA_M2M = DMA_M2M_Enable;   //选择是硬件触发还是软件触发    软件触发
	DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;   // 优先级    选择中等优先级
	DMA_Init(DMA1_Channel1, &DMA_InitStruct);  //第一个参数选择了是哪个DMA到哪个DMA通道,第二个参数结构体
	
	//第三步:调用DMA_CMD,通道使能(要在对应的外设调用XXX_DMACmd开启一下触发信号的输出,如果需要DMA的中断,就调用DMA_ITConfig,开启中断输出,再在NVIC里,配置中断通道,最后写中断函数就行了)
	DMA_Cmd(DMA1_Channel1, ENABLE);
}







下面是MyDMA.h文件:

#ifndef __MYDMA_H
#define __MYDMA_H


void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size);



#endif

下面是main.c文件:

#include "stm32f10x.h"                  // Device header
#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_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数据搬运的小功能了。看似没有什么用,其实这功能是在硬件上自动完成的,可以在不浪费软件资源的前提下,把外设上的数据搬运到内存中来,软件就能直接用了,还是非常的方便的。

今天又更改了一点代码,单独写了一个函数用来启动搬运工作,更改位置为:

STM32实验DMA数据搬运小助手,STM32,stm32,嵌入式硬件,单片机

源文件为:

#include "stm32f10x.h"                  // Device header

uint16_t MyDMA_Size;

void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{
	MyDMA_Size = Size;
	//1,RCC
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	
	//2,DMA_Init()
	DMA_InitTypeDef DMA_InitStruct;
	DMA_InitStruct.DMA_PeripheralBaseAddr = AddrA;
	DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
	DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Enable;
	
	DMA_InitStruct.DMA_MemoryBaseAddr = AddrB;
	DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
	DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
	
	DMA_InitStruct.DMA_BufferSize = Size;
	DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;
	DMA_InitStruct.DMA_M2M = DMA_M2M_Enable;
	DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
	DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;
	DMA_Init(DMA1_Channel1, &DMA_InitStruct);
	//3:DMA_Cmd
	DMA_Cmd(DMA1_Channel1, DISABLE);
}

void DMA_BanYun(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);
}

下面就是主main函数的内容了:

STM32实验DMA数据搬运小助手,STM32,stm32,嵌入式硬件,单片机

这样就会直接显示数组A和B的每一个元素的内容和数组的地址,随后数组A中的每一个元素都自加1,屏幕是可以计时看到结果的,过1秒钟后开始搬运,才能看到搬运的结果,数组B的值发生变化,就这样周而复始的循环。来个图片看看吧,CSDN上放个小视频太麻烦了,也不改进一下???下面图片中的数值是在不断地自增中。

STM32实验DMA数据搬运小助手,STM32,stm32,嵌入式硬件,单片机文章来源地址https://www.toymoban.com/news/detail-845197.html

到了这里,关于STM32实验DMA数据搬运小助手的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 嵌入式开发:使用STM32CubeMX 和 Keil uVision5实现两位数码管循环显示实验

    一、STM32CubeMx实现数码管 1、双击打开STM32CubeMX。 2、点击File选项,在File列表下选择New Project选项新建工程。 3、在图中所示位置的方框中输入我们使用的芯片型号来搜索芯片 4、与我们的搜索相关的芯片型号会在这里呈现,我们只需要选择我们需要的即可,这里我们双击第一个

    2024年02月08日
    浏览(54)
  • STM32嵌入式系统:将数据保存到SD卡的操作

    STM32嵌入式系统:将数据保存到SD卡的操作 嵌入式系统在现代科技中扮演着重要角色,而STM32单片机是一种常用的嵌入式系统解决方案。本文将介绍如何使用STM32单片机将采集到的数据以TXT文件的格式保存到SD卡中,并且能够方便地读取这些本地数据。 硬件准备 为了实现数据保

    2024年02月01日
    浏览(45)
  • 【嵌入式学习笔记】嵌入式基础9——STM32启动过程

    程序段交叉引用关系(Section Cross References):描述各文件之间函数调用关系 删除映像未使用的程序段(Removing Unused input sections from the image):描述工程中未用到被删除的冗余程序段(函数/数据) 映像符号表(Image Symbol Table):描述各符号(程序段/数据)在存储器中的地址、类

    2024年02月15日
    浏览(87)
  • STM32的时钟系统(嵌入式学习)

    时钟是指用于计量和同步时间的装置或系统。时钟是嵌入式系统的脉搏,处理器内核在时钟驱动下完成指令执行,状态变换等动作,外设部件在时钟的驱动下完成各种工作,例如:串口数据的发送、AD转换、定时器计数等。因此时钟对于计算机系统是至关重要的,通常时钟系

    2024年02月16日
    浏览(50)
  • 嵌入式——新建STM32工程(标准库)

    目录 一、初识标准库 1.CMSIS标准及库层级关系 2.库文件介绍 (1)Libraries文件夹 ①CMSIS文件夹 ②STM32F10x_Std_Periph_Driver文件夹 ③ 在用库建立一个完整的工程时,还需要添加stm32f10x_it.c、 stm32f10x_conf.h 和 system_stm32f10x.c文件 (2)Project文件夹 (3)Utilities文件夹 3.库各文件之间的关

    2024年01月23日
    浏览(59)
  • STM32串口通信详解(嵌入式学习)

    时钟信号在电子领域中是指用于同步和定时电路操作的周期性信号。它在数字系统和通信系统中起着至关重要的作用,用于协调各个组件之间的数据传输和操作。 时钟信号有以下几个重要的方面: 频率:时钟信号的频率是指单位时间内信号周期的数量。它通常以赫兹(Hz)为

    2024年02月09日
    浏览(69)
  • 嵌入式 STM32 通讯协议--MODBUS

    目录 一、自定义通信协议 1、协议介绍 2、网络协议 3、自定义的通信协议  二、MODBUS通信协议 1、概述 2、MODBUS帧结构  协议描述 3、MODBUS数据模型   4、MODBUS事务处理的定义 5、MODBUS功能码  6、功能码定义   7、MODBUS数据链路层 8、MODBUS地址规则  9、MODBUS帧描述 10、MODBUS两种

    2024年02月11日
    浏览(67)
  • stm32毕设分享 Stm32酒驾检查系统 - 单片机 嵌入式 物联网

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年01月23日
    浏览(76)
  • stm32毕设分享 stm32实现车牌识别系统 -物联网 嵌入式 单片机

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月20日
    浏览(63)
  • 嵌入式学习笔记——STM32的时钟树

    在之前的所有代码编程的过程中,似乎每次都绕不开一个叫做时钟使能的东西,当时我们是在数据手册上直接看其挂接在那条时钟线上的,那么STM32内部的时钟到底是怎么一个构型呢,本文来对此做一个介绍。 老规矩,一个新的名词出现,首先需要搞清楚它是个啥,下图中对

    2024年02月02日
    浏览(58)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包