STM32F4关于DMA传输向GPIO口的开发

发布者:创新之梦最新更新时间:2021-02-20 来源: eefocus关键字:STM32F4  DMA传输  GPIO口 手机看文章 扫描二维码
随时随地手机看文章

本文章是经历了大量时间,试验,阅读文档,上网搜索无果,再读文档。最后,睡觉时做了一个梦,在梦中,对文档从头到尾再过了一遍,第二天早上醒来,按照梦中的指示,做了些许修改,一次出结果的。


希望此文能够帮助到国内还在此问题上困扰的人们。


说到STM32的DMA,其实大家都已经很熟悉了。DMA的例子网上也是到处都有。在F1的开发中,DMA需要设置的就是这些内容了,理解上很容易。 主要就是:


1.设置通道


2.设置源地址和目标地址


3.设置buffer长度


4.设置方向


5.设置模式


6.设置各地址的自增特性


7.设置传输字长


8.设置搬运模式,单次,循环


9.设置优先级


在F4上还增加了设置FIFO


至于代码,大家根据上面的过程参考各自的代码,关键不在这里。


本文的重点是实现DMA向GPIO的传输


在F103中写法:


DMA_DeInit(DMA1_Channel6);

 

DMA_InitStructure.DMA_PeripheralBaseAddr =  (uint32_t)(&(GPIOB->ODR));  

DMA_InitStructure.DMA_MemoryBaseAddr =(u32) &aa[0]

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;

DMA_InitStructure.DMA_BufferSize = num;

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;

DMA_InitStructure.DMA_PeripheralDataSize =DMA_PeripheralDataSize_Byte;

DMA_InitStructure.DMA_MemoryDataSize =DMA_MemoryDataSize_Byte;

DMA_InitStructure.DMA_Mode =DMA_Mode_Normal;//需要软件启动

DMA_InitStructure.DMA_Priority = DMA_Priority_Low;

DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;

DMA_Init(DMA1_Channel6, &DMA_InitStructure);

上面这个是向GPIOB的pin0-pin7脚单次传输一个字节


而在F4中的写法:


DMA_DeInit(DMA2_Stream5);

 

DMA_InitStructure.DMA_Channel = DMA_Channel_5;               /* 配置DMA通道 */

DMA_InitStructure.DMA_PeripheralBaseAddr =(uint32_t) &aa[0];   // 

DMA_InitStructure.DMA_Memory0BaseAddr =(uint32_t)&(GPIOB->ODR); 

DMA_InitStructure.DMA_DIR =DMA_DIR_MemoryToMemory;

DMA_InitStructure.DMA_BufferSize = num;

DMA_InitStructure.DMA_PeripheralInc =DMA_PeripheralInc_Enable;

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;

DMA_InitStructure.DMA_PeripheralDataSize =DMA_PeripheralDataSize_Byte;

DMA_InitStructure.DMA_MemoryDataSize =DMA_MemoryDataSize_Byte;

DMA_InitStructure.DMA_Mode =DMA_Mode_Normal

DMA_InitStructure.DMA_Priority = DMA_Priority_Low;

  DMA_InitStructure.DMA_FIFOMode    = DMA_FIFOMode_Disable; /*直接模式 */

    DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_3QuartersFull; /* FIFO大小 */

    DMA_InitStructure.DMA_MemoryBurst    = DMA_MemoryBurst_Single;       /* 单次传输 */

    DMA_InitStructure.DMA_PeripheralBurst= DMA_PeripheralBurst_Single;

 

DMA_Init(DMA2_Stream5, &DMA_InitStructure);

 

,咦,怎么可能呢。源地址和目标地址竟然对调了!!!这能行吗???


首先:从DMA传向GPIO属于M2M模式,切记!,不管是F1还是F4或其他的。




其次,F4只有DMA2才可以做上述功能。


第三,F4方向有不同!不多说,看文档:

看到了吗,在F4上,M2M时,源地址和目标地址的寄存器是反过来的,相应的要把赋值也反过来,内存自增幅值也要反过来。

到此,信号已经能正常从GPIO口输出,对经历的过程予以总结,同时希望同行看到此文可以有所收获。


关键字:STM32F4  DMA传输  GPIO口 引用地址:STM32F4关于DMA传输向GPIO口的开发

上一篇:STM32F4的GPIO口配置模式
下一篇:STM32f4 GPIO的基础使用-流水灯

推荐阅读最新更新时间:2024-11-17 06:26

NRF24L01调试记 (STM32F4,STM32F1,STM8L)
其实网上NRF24L01的资料很多,例程也很多,本不应该有什么大问题,但是确出了各种奇怪的问题,花了不少时间。 STM32F103+NRF24L01 用最小系统板搭了个平台,加上网上的例程,问题是检查NRF24L01的时候,一直没法找到,读写地址寄存器失败,起初怀疑是引线的问题,自己检查连线以及针脚定义,都没有发现问题,既然针脚没有问题怀疑是SPI通讯的问题,接着检查SPI配置,都没有发现明显的错误,尽管分频到256还是不行,十分不解,检查时序代码,因为是网上现成的代码本来想不应该有问题,但是还真的发现问题了,发现网上时序先是将CSN拉高!发送结束后再拉低!这个明显不符,修改后运行正常,哎,网上人家给的例程未必人家就验证过,切
[单片机]
NRF24L01调试记 (<font color='red'>STM32F4</font>,STM32F1,STM8L)
STM32F407 TIM1的PWM输出
//TIM1 PWM部分初始化 //PWM输出初始化 //arr:自动重装值 //psc:时钟预分频数 void TIM1_PWM_Init(u32 arr,u32 psc) { //此部分需手动修改IO口设置 GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1,ENABLE); //TIM1时钟使能 RCC_AHB1P
[单片机]
STM32F407ADC多通道+定时器触发+DMA模式设置
#include adc.h u8 UpdataTIM = 0; //更新TIM2定时器标志 u16 CurrentFreq = 500;//默认市电频率,单位为0.1hz ADBASE_TYPE uAD_Buff_A ; //ADC采集缓存A ADBASE_TYPE uAD_Buff_B ; //ADC采集缓存B //考虑到可能出现的数据处理时间较长的问题,临时添加缓存 //u16 uAD_IN_BUFF ; ADBASE_TYPE *CurrentBuffPtr = NULL; //当前缓存指针 static void ADCInit_GPIO(void) { GPIO_InitTypeDef G
[单片机]
STM32F407--GPIO的工作原理
一、参考资料 1、STM32F407ZGT6.pdf(探索者资料盘A7,硬件资料2,芯片资料) 2、STM32F4xx中文参考手册.pdf(探索者资料盘A8,STM32参考资料) 3、STM32F4开发指南-库函数版本_V1.1.pdf(探索者资料盘A) 二、GPIO口的基本结构 三、引脚的说明 注:STM32大部分引脚可以当GPIO口外,还可以复用为外设功能引脚(如串口)。 四、GPIO的工作模式 1、4种输入模式 (1)GPIO_Mode_IN_FLOATING 浮空输入 (2)GPIO_Mode_IPU 上拉输入 (3)GPIO_Mode_IPD 下拉输入 (4)GPIO_Mode_AI
[单片机]
<font color='red'>STM32F4</font>07--<font color='red'>GPIO</font>的工作原理
SPI外设与USART外设之间通过DMA直接传输初步
选取DMA1 选取通道2 SPI1 USART3 根据上面这些配置初始化,参考32例程。 下面是我配置的SPI中断,我把它直接放在SPI初始化函数后面 先中断初始化 然后建立外设之间的连接 void SPI1_IRQHandler(void) USART3- DR=SPI1- DR; 再直接进行地址映射,把SPI1外设直接映射到USART3。 使能SPI的DMA发送请求PI_I2S_DMACmd(SPI1, SPI_I2S_DMAReq_Tx|SPI_I2S_DMAReq_Rx, ENABLE); 就会启动DMA 主函数
[单片机]
SPI外设与USART外设之间通过<font color='red'>DMA</font>直接<font color='red'>传输</font>初步
STM32F429 DMA串口数据发送
串口数据配置 void macUART4_Config(void) { GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; RCC_AHB1PeriphClockCmd(macUART4_RX_GPIO_CLK|macUART4_TX_GPIO_CLK,ENABLE); RCC_APB1PeriphClockCmd(macUART4_CLK, ENABLE); GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Pu
[单片机]
STM32采集AD电压
是否使用DMA传输 使用DMA传输,那么流程为: ADC初始化,DMA初始化,TIM2初始化 其中:TIM2作为ADC的中断源 当发生一次定时器的中断时,进入AD转换,在DMA的初始化时与ADC-DR寄存器进行绑定,在该寄存器获得数据时,直接通过DMA通道将该寄存器的数据保存在给定的数组里面,把缓存数组装满后,会触发一次DMA的中断,在DMA的中断里面将缓存数组保存到100个电压值的数组里面。 定时器设为200us发生一次中断,来进行一次AD转换,获得寄存器里面的数据以后,保存在数组,保存100个数组后停止定时器的工作以及DMA的工作,对这份数据进行处理。 至此,DMA工作的流程已经结束,那么其中又怎样的缺漏导致不用DMA进行
[单片机]
STM32 DMA正常模式等待传输完成和开始下一次传输
选择DMA的正常模式,即DMA只传输一次。如果当传输完一次后,还想再传输一次,就需要重启DMA: DMA_Cmd(DMA1_Channel6,DISABLE); 重新设置源地址 重新设置目的地址 重设传输数量 DMA_Cmd(DMA1_Channel6,ENABLE); 因此建议把DMA_InitStructure定义为全局变量,然后可以使用下面语句来重启DMA: DMA_Cmd (DMA1_Channel4,DISABLE); DMA_Init(DMA1_Channel4, &DMA_InitStructure); DMA_Cmd (DMA1_Channel4,ENABLE); 刚发现只需要重新设置传输数据长度就可以
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

更多开源项目推荐
换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved