【stm32】DMA的介绍与使用

news2025/1/14 19:21:28

DMA的介绍与使用

  • 1、DMA简介
  • 2、存储器映像
  • 3、DMA框图
  • 4、DMA基本结构
  • 5、DMA请求
  • 6、数据宽度与对齐
  • 7、数据转运+DMA(存储器到存储器的数据转运)
    • 程序编写:
  • 8、ADC连续扫描模式+DMA循环转运
    • DMA配置:
    • 程序编写:

1、DMA简介

  • DMA(Direct Memory Access)直接存储器存取
    • 可以直接访问STM32内部存储器,包括运行内存SRAM、程序存储器Flash和寄存器等
  • DMA可以提供外设和存储器或者存储器和存储器之间的高速数据传输,无须CPU干预,节省了CPU的资源
    • 外设:指的是外设寄存器(一般是外设数据寄存器DR,如ADC的数据寄存器、串口的数据寄存器)
    • 存储器:指运行内存SRAM和程序存储器Flash(存储变量数组和程序代码)
  • 12个独立可配置的通道: DMA1(7个通道), DMA2(5个通道)
  • 每个通道都支持软件触发和特定的硬件触发
    • 软件触发:一般使用存储器到存储器
    • 硬件触发:一般使用外设到存储器
  • STM32F103C8T6 DMA资源:DMA1(7个通道)

2、存储器映像

在这里插入图片描述
计算机系统的5大组成部分:运算器、控制器、存储器、输入设备和输出设备
运算器、控制器统称为CPU

3、DMA框图

在这里插入图片描述

  • DMA总线:用于访问各个存储器,内部多个通道,可以进行独立的数据转运
  • 仲裁器:用于调度各个通道,防止产生冲突
  • AHB从设备:用于配置DMA参数
  • DMA请求:用于硬件触发DMA的数据转运

4、DMA基本结构

在这里插入图片描述

  • 起始地址:外设端的起始地址和存储器端的起始地址(决定数据的方向)
  • 数据宽度:指定一次转运要按多大的数据宽度来进行(字节Byte[8位]、半字HalfWord[16位]和字Word[32位])
  • 地址是否自增:指定一次转运完成后,下一次转运,是不是要把地址移动到下一个位置去
  • 传输计数器:自减计数器,计数器值即DMA转运的次数 ,减到0后,自增的地址会恢复到起始地址的位置,以方便之后DMA开始新一轮的转运
  • 自动重装器:决定转运的模式,不重装为单次模式,重装为循环模式
    • 传输计数器减到0之后,是否要自动恢复到最初的值,如传输计数器值为5,如果不使用自动重装器,转运5次后,DMA转运结束,如果使用自动重装器,转运5次,计数器减到0后,会立即重装到初始值5,DMA再次开始转运数据
  • 触发控制:决定DMA需要在什么时机进行转运。
    • M2M(Memory to Memory)决定选择哪个触发源,M2M = 1,DMA选择软件触发(ps:软件触发和循环模式不能同时使用,因为软件触发就是把传输计数器清零,循环模式是清零后自动重装,如果同时使用,DMA无法停止);M2M = 0,DMA选择硬件触发(ADC,串口,定时器),硬件触发的转运,一般都与外设有关的转运,这些转运需要一定的时机,比如ADC转换完成、串口收到数据、定时时间到等,在硬件达到这些时机时,传一个信号过来,触发DMA进行转运。
  • 开关控制:调用DMA_Cmd函数,当给DMA使能后,DMA准备就绪,可以进行转运
    • DMA可以转运的条件:1、开关控制,DMA_Cmd必须使能;2、传输计数器必须大于0;3、触发源,必须有触发信号

注意:写传输计数器时,必须要先关闭DMA,再进行

5、DMA请求

在这里插入图片描述
DMA触发部分:
EN位:EN=0,数据选择器不工作,EN=1,数据选择器工作
软件触发后面跟个M2M位的意思:当M2M位=1时,选择软件触发

PS:想使用某个硬件触发源,必须使用它所在的通道;使用软件触发,通道可以任意选择

6、数据宽度与对齐

在这里插入图片描述
源端宽度比目标宽度小时:

  • 当源端和目标都是8位时,转运第一步,在源端的0位置,读数据B0,在自标的0位置,写数据B0,即把B0,从左边挪到右边,后续B1,B2,B3同理;
  • 当源端是8位,目标是16位,在源端读B0,在目标写00B0,之后,读B1,写00B1等,后续同理
    • 总结:若目标的数据宽度比源端的数据宽度大,在目标数据前面多出来的空位补0
  • 当源端是8位,目标是32位,同源端8位,目标16位处理方式相同

源端宽度比目标宽度大时:

  • 当源端是16位,目标是8位,读B1BO,只写入B0;读B3B2,只写入B2,
    • 总结:将高位舍弃
  • 其它方式依次类推

7、数据转运+DMA(存储器到存储器的数据转运)

在这里插入图片描述

将SRAM里的数组DataA,转运到另一个数组DataB中

  • 外设地址:DataA数组首地址 | 存储器地址:DataB数组的首地址
  • 数据宽度:两个数组的类型都是uint8_t,按8位的字节传输
  • 地址自增:数组DataA和DataB均自增
  • 转运方向:存储器向存储器转运
  • 传输计数器;转运7次,自动重装器不使用
  • 触发控制:软件触发

程序编写:

// myDMA.c
#include "stm32f10x.h"                  // Device header
/*
DMA 转运的3个条件
1、传输计数器大于0      Size > 0
2、触发源有触发信号 
3、DMA使能
*/

uint16_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_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;                     // 传输模式,是否使用自动重装--不能和软件触发同时使用,选择不自动重装,计数减到0后停止
    DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;                     // 选择是否是存储器到存储器,即选择硬件触发还是软件触发
    DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;             // 优先级
    DMA_Init(DMA1_Channel1, &DMA_InitStructure);
    
    DMA_Cmd(DMA1_Channel1, DISABLE); // 初始化不立刻转运,使用MyDMA_Transfer函数进行转运
}
// 调用此函数,再次启动一次DMA转运
void MyDMA_Transfer(void)
{
    // 重新给传输计数器赋值,先把DMA失能
    DMA_Cmd(DMA1_Channel1, DISABLE);        // DMA失能
    DMA_SetCurrDataCounter(DMA1_Channel1, MyDMA_Size); // 传输计数器赋值
    DMA_Cmd(DMA1_Channel1, ENABLE);     // DMA 使能
    
    while (DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);// 等待转运完成
    DMA_ClearFlag(DMA1_FLAG_TC1);// 清除标志位
}
// 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();
    // DataA地址数据转运到DataB地址
    MyDMA_Init((uint32_t)DataA, (uint32_t)DataB, 4);
    
    OLED_ShowString(1, 1, "DataA");
    OLED_ShowString(3, 1, "DataB");
    OLED_ShowHexNum(1, 8, (uint32_t)DataA, 8);
    OLED_ShowHexNum(3, 8, (uint32_t)DataB, 8);
        
    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);
    }
}

8、ADC连续扫描模式+DMA循环转运

在这里插入图片描述

左边为ADC扫描模式的执行流程:7个通道,触发一次后,7个通道依次进行AD转换,转换结果都放到ADC_DR数据寄存器中,需要在每个单独的通道转换完成后,进行一个DMA数据转运,并且目的地址进行自增,防止数据被覆盖。

DMA配置:

  • 外设地址写入ADC_DR寄存器的地址,存储器地址,可以在SRAM中定义一个数组ADValue,把ADValue的地址当做存储器的地址
  • 数据宽度:16位的半字传输
  • 地址是否自增;外设地址不自增,存储器地址自增
  • 传输方向:外设到存储器
  • 传输计数器:通道有7个,计数7次
  • 计数器是否自动重装,看ADC的配置,ADC如果是单次扫描,DMA的传输计数器可以不自动重装,如果ADC是连续扫描,DMA可以使用自动重装,在ADC启动下一轮转换时,DMA也启动下一轮转运,ADC和DMA同步工作
  • 触发选择:ADC_DR的值是在ADC单个通道转换完成后才会有效,所以DMA转运的时机,需要和ADC单个通道转换完成同步,DMA的触发要选择ADC的硬件触发

程序编写:

// AD.c
// 使用软件触发ADC采集数据--->ADC硬件触发DMA转运数据到存储器
#include "stm32f10x.h"                  // Device header

uint16_t AD_Value[4];

// ADC连续扫描模式+DMA循环转运
void AD_Init(void)
{
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
    
    RCC_ADCCLKConfig(RCC_PCLK2_Div6);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    
    ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);
    ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5);
    ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5);
    ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5);
        
    ADC_InitTypeDef ADC_InitStructure;
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; // 连续扫描
    ADC_InitStructure.ADC_ScanConvMode = ENABLE;  // 使用扫描模式
    ADC_InitStructure.ADC_NbrOfChannel = 4;
    ADC_Init(ADC1, &ADC_InitStructure);
    
    DMA_InitTypeDef DMA_InitStructure;
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR; // ADC DR的基地址,ADC采集到的数据存到此寄存器中
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; // 16位
    DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;// 外设地址不需自增
    DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_Value; // 转运数据到的目的地址
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; // 16位
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  // 存储器地址自增
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
    DMA_InitStructure.DMA_BufferSize = 4;  //  传输次数-4个ADC通道
    DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; // 循环模式
    DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; // 硬件触发
    DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
    DMA_Init(DMA1_Channel1, &DMA_InitStructure); // ADC 硬件触发只在DMA1的通道1上
    
    DMA_Cmd(DMA1_Channel1, ENABLE);
    ADC_DMACmd(ADC1, ENABLE); // 开启ADC到DMA的输出
    ADC_Cmd(ADC1, ENABLE);
    
    ADC_ResetCalibration(ADC1);
    while (ADC_GetResetCalibrationStatus(ADC1) == SET);
    ADC_StartCalibration(ADC1);
    while (ADC_GetCalibrationStatus(ADC1) == SET);
    
    ADC_SoftwareStartConvCmd(ADC1, ENABLE);//ADC开始工作
}
//main.c
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

int main(void)
{
    OLED_Init();
    AD_Init();
    
    OLED_ShowString(1, 1, "AD0:");
    OLED_ShowString(2, 1, "AD1:");
    OLED_ShowString(3, 1, "AD2:");
    OLED_ShowString(4, 1, "AD3:");
    
    while (1)
    {
        OLED_ShowNum(1, 5, AD_Value[0], 4);
        OLED_ShowNum(2, 5, AD_Value[1], 4);
        OLED_ShowNum(3, 5, AD_Value[2], 4);
        OLED_ShowNum(4, 5, AD_Value[3], 4);
        
        Delay_ms(100);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2215014.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Antsword-labs靶机渗透

Less-1 在当前界面开启终端 ![](https://img-blog.csdnimg.cn/img_convert/e5ab1b947b1186a43b58abaf10263cb1.png) 启动环境 plain docker-compose up -d 蚁剑连接 ![](https://img-blog.csdnimg.cn/img_convert/81a5c09987e18355335d07e4da52cb5f.png) 打开终端寻找flag …

ACL:访问控制列表

基本概念 1.访问控制: 在路由器的入或者出的接口上,匹配流量,之后产生动作,只有允许和拒绝。 2.定义感兴趣流量: 帮助其他策略抓流量的 匹配规则:至上而下 逐一匹配 上条匹配按照上条执行 不再查看下条 (思科体系中 末尾隐含拒绝所有 华为…

20个月投标战胜国际对手,中国百余台AGV进驻欧洲……

导语 大家好,我是社长,老K。专注分享智能制造和智能仓储物流等内容。 在全球智能制造竞争日益激烈的今天,中国制造再次传来振奋人心的好消息。 经过长达20个月的艰苦角逐,一家中国机器人企业(新松机器人)成…

microsoft edge浏览器卡死问题

win11经常遇到microsoft edge浏览器卡死的情况,有时候是一会没用浏览器就全部卡死,有时候是锁屏或者电脑休眠浏览器就不能用,找了很多的办法都没好使,用以下方法好使了: edge浏览器中打开 edge://settings/system 把 …

计算机的错误计算(一百二十三)

摘要 探讨算式 的计算精度问题。 例1. 已知 计算 不妨在Python下计算,则有: 若用Rust在线计算: fn main() {let x: f64 0.125e-6;let tan_x x.tan();let sin_x x.sin();let denominator x - (1.0 x * x).sqrt();let result (…

大学模拟电路设计期末速成总结

大学模拟电路设计期末速成总结 模拟电路设计是电子工程领域的基础,它涉及到连续信号的处理和放大。对于电子工程的学生来说,掌握模拟电路设计的基本原理和应用是至关重要的。以下是期末速成总结,帮助你快速回顾和掌握模拟电路设计的关键知识…

香橙派刷机和开发环境准备(ubuntu22.04版)_随记1

前言: 一、香橙派刷ubuntu系统和SSH登录 1、刷机前准备: ①TF卡(8G)、读卡器、OrangePi5Pro ②Win32DiskImager(烧写系统工具)、SDFormatter(TF格式化工具) ③系统镜像&#xff…

学习文档(4)

目录 Vue简介 MVVM思想 Vue指令 内容输出指令 条件渲染指令 列表渲染指令 数据绑定指令 Vue简介 Vue2.x官网:https://v2.cn.vuejs.org Vue3.x官网:https://cn.vuejs.org 官网(2.x版本)对vue的定义是:vue是一套…

kubelet PLEG实现

概述 kubelet的主要作用是确保pod状态和podspec保持一致,这里的pod状态包括pod中的container状态,个数等。 为了达到这个目的,kubelet需要从多个来源watch pod spec的变化,并周期从container runtime获取最新的container状态。比如…

I2C相关结构体讲解:i2c_adapter、i2c_algorithm、i2c_msg

往期内容 I2C子系统专栏: I2C(IIC)协议讲解SMBus 协议详解 总线和设备树专栏: 专栏地址导航篇 – 专栏未篇 1.框图 建议右击图片在新标签页打开预览 i2c_transfer函数就是读取i2c设备的信息或者输出信息给i2c设备的函数 比如发送…

信息安全工程师(49)网络物理隔离系统与类型

前言 网络物理隔离系统是指通过物理隔离技术,在不同的网络安全区域之间建立一个能够实现物理隔离、信息交换和可信控制的系统,以满足不同安全域的信息或数据交换需求。 一、网络物理隔离系统概述 网络物理隔离系统的核心在于通过物理方式将不同安全级别的…

SQL Injection | SQL 注入概述

关注这个漏洞的其他相关笔记:SQL 注入漏洞 - 学习手册-CSDN博客 0x01:SQL 注入漏洞介绍 SQL 注入就是指 Web 应用程序对用户输入数据的合法性没有判断,前端传入后端的参数是可控的,并且参数会带入到数据库中执行,导致…

【计算机网络篇】数据链路层 协议、介质访问控制

🧸安清h:个人主页 🎥个人专栏:【计算机网络】【Mybatis篇】 🚦作者简介:一个有趣爱睡觉的intp,期待和更多人分享自己所学知识的真诚大学生。 系列文章目录 【计算机网络篇】计算机网络概述 【…

[实时计算flink]CREATE DATABASE AS(CDAS)语句

CDAS支持整库级别的表结构和数据的实时同步,还支持表结构变更的同步。本文为您介绍CREATE DATABASE AS(CDAS)的使用方法,并提供了多种使用场景下的示例。 背景信息 CDAS是CTAS语法的一个语法糖,用于实现整库同步、多…

在 Unity 中创建模型动画的探索之旅

在 Unity 游戏开发或 3D 场景构建中,模型动画是赋予虚拟对象生命和个性的关键元素。它能够极大地增强用户体验,使场景更加生动和吸引人。本文将带您深入了解在 Unity 中创建模型动画的基本流程和方法。 一、准备工作 在开始创建动画之前,您…

空间大数据的数据变换与价值提炼

在数字化时代,空间大数据正成为推动社会经济发展的关键因素。空间大数据不仅体量巨大,而且具有高速流转、多样类型和真实性等特点,它们在获取、存储、管理、分析方面超出了传统数据库软件工具的能力范围。地理信息系统(GIS&#x…

淘宝详情API接口有什么用处?

淘宝详情API接口有什么用处?主要体现在以下几个方面: 电商数据分析:通过调用API接口获取商品详情数据,可以对商品的销售情况、价格变化、属性分布等进行深入分析。这些数据为电商运营提供了决策支持,帮助商家更好地了…

Redis哨兵模式部署(超详细)

哨兵模式特点 主从模式的弊端就是不具备高可用性,当master挂掉以后,Redis将不能再对外提供写入操作,因此sentinel模式应运而生。sentinel中文含义为哨兵,顾名思义,它的作用就是监控redis集群的运行状况,此…

使用gradle将java项目推送至maven中央仓库(最新版)

前言 maven中央仓库于2024年3月进行改版,下面介绍新的推送方式 一、将项目推送到github 过程略 二、注册sonatype账号 仓库地址:https://central.sonatype.com/ 这里选择使用github账号登录,不注册新的了 三、创建命名空间 这里会自动…

2012年国赛高教杯数学建模D题机器人避障问题解题全过程文档及程序

2012年国赛高教杯数学建模 D题 机器人避障问题 图1是一个800800的平面场景图,在原点O(0, 0)点处有一个机器人,它只能在该平面场景范围内活动。图中有12个不同形状的区域是机器人不能与之发生碰撞的障碍物,障碍物的数学描述如下表&#xff1a…