关于STM32F407的ADC+TIM+DMA采样

古城旧巷闻少年 2017-08-24 08:08:22
不知道哪里出问题了,初衷是用TIM3每500ms触发一次ADC1的通道5采样外部电压,再通过DMA传数据到内存,但是我调试时发现 ADC1的数据寄存器只更新的一次,就卡在while(1)里面了。
//adc.h

#ifndef __ADC__
#define __ADC__
#include "sys.h"

void gpioled_config(void);
void gpio_config(void);
void adc_config(void);
void tim_config(void);
void dma_config(void);

#endif
*********************************************************************************************************

//adc.c

#include "stm32f4xx_conf.h"
#include "sys.h"
#include "stm32f4xx_spi.h"
#include "delay.h"
#include "stm32f4xx_adc.h"
#include "stm32f4xx_dma.h"
#include "stm32f4xx_dma2d.h"
#include "adc.h"


void gpioled_config(void)
{
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF, ENABLE);

GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOF, &GPIO_InitStructure);
}

void gpio_config(void)
{
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);

GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void adc_config(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);

ADC_CommonInitTypeDef ADC_CommonInitStructure;
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //是否提高采样率
ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent; //ADC独立模式
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4; //配置成21M,保证ADC不超过36M
ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles; //采样通道之间的间隔
ADC_CommonInit(&ADC_CommonInitStructure);

ADC_InitTypeDef ADC_InitStructure;
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //ADC单次模式
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //数据右对齐
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T3_TRGO; //TIM3外部触发
ADC_InitStructure.ADC_ExternalTrigConvEdge =ADC_ExternalTrigConvEdge_Rising; //上升沿触发
ADC_InitStructure.ADC_NbrOfConversion = 1; //一个转换序列
ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b; //采样精度为12位
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //扫描模式禁止
ADC_Init(ADC1,&ADC_InitStructure);

ADC_RegularChannelConfig(ADC1,ADC_Channel_5,1,ADC_SampleTime_480Cycles); //配置ADC转换通道

ADC_Cmd(ADC1,ENABLE); //使能ADC
ADC_DMACmd(ADC1,ENABLE); //使能ADC的DMA
}

void tim_config(void)
{
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);

TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV4; //定时器时钟168M/4*2 = 84M
TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数模式
TIM_TimeBaseInitStructure.TIM_Period = 4999; // 重载数
TIM_TimeBaseInitStructure.TIM_Prescaler = 8399; //分频系数
// TIM_TimeBaseInitStructure.TIM_RepetitionCounter 高级定时器才用到,不管
TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);

TIM_ARRPreloadConfig(TIM3,ENABLE);

TIM_SelectOutputTrigger(TIM3,TIM_TRGOSource_Update); //定时器触发输出

TIM_Cmd(TIM3,ENABLE); // 使能定时器
}

void dma_config(void)
{
u16 adcbuffer[24];
u16 *a = adcbuffer;

RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1,ENABLE);

DMA_DeInit(DMA1_Stream4);

DMA_InitTypeDef DMA_InitStructure;
DMA_InitStructure.DMA_BufferSize = 128;
DMA_InitStructure.DMA_Channel= DMA_Channel_5;//DMA通道5
DMA_InitStructure.DMA_DIR= DMA_DIR_PeripheralToMemory;//传输方向为外设到内存
DMA_InitStructure.DMA_FIFOMode= DMA_FIFOMode_Disable;//DMA的FIFO模式禁止
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
DMA_InitStructure.DMA_Memory0BaseAddr = ((uint16_t)&adcbuffer); //内存地址
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //半字传输
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址递增
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //DMA循环模式开启
DMA_InitStructure.DMA_PeripheralBaseAddr = ((uint32_t)&(ADC1->DR)); //外设地址
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//半字传输
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址递增
DMA_InitStructure.DMA_Priority = DMA_Priority_High;//DMA优先级高
DMA_Init(DMA1_Stream4, &DMA_InitStructure);

DMA_Cmd(DMA1_Stream4,ENABLE);
}

*******************************************************************************************************************

//mian.c

#include "stm32f4xx.h"
#include "usart.h"
#include "delay.h"
#include "adc.h"


int main(void)
{
uart_init(115200);
delay_init(168);
gpioled_config();
PFout(9) = 0;
PFout(10) = 0;
gpio_config();
adc_config();
dma_config();
tim_config();

while(1)
{
PFout(9) = 1;
PFout(10) = 1;
delay_ms(100);
PFout(9) = 0;
PFout(10) = 0;
delay_ms(100);
}
}

**************************************************************************************************
以上就是我的程序了,是哪里没配,还是哪里没配对,请大家多多指教!!!


...全文
2452 11 打赏 收藏 转发到动态 举报
写回复
用AI写文章
11 条回复
切换为时间正序
请发表友善的回复…
发表回复
m0_45697066 2021-03-05
  • 打赏
  • 举报
回复
发现问题了,是内存地址应该设置成不递增的,不然容易溢出,失能后就正常了。
m0_45697066 2021-03-05
  • 打赏
  • 举报
回复
楼主现在解决了吗?F4VET6同款型号,同样的问题,DMA数据只传送一次
qq_41811627 2019-11-16
  • 打赏
  • 举报
回复


好像每次DMA传输完成都会导致溢出。暂时这样解决
Wolfcity 2018-03-16
  • 打赏
  • 举报
回复
是不是还应该实现TIM定时中断啊,重新装载计数值。。。
nettman 2017-08-26
  • 打赏
  • 举报
回复
关注
gaofto 2017-08-25
  • 打赏
  • 举报
回复
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //ADC单次模式 这句话表明的应该就是单次转换吧? 如果不是这句话,那就看看是不是定时器是单次定时,还是重载定时? 转换一次就停止的话,问题一般都是这两个
  • 打赏
  • 举报
回复
谢谢1、2L的回答。1L大大原理说的很透彻。
正如2L所说,现在问题就是用TIM只触发了一次ADC采样。
采用TIM+ADC+DMA是为了解放CPU,提高工作效率。DMA工作的前提是外设有数据让它取,就是ADC数据寄存器ADC_DR中有值更新,但是我用Keil5调试时发现ADC_DR中的值只更新了一次,这代表TIM定时到后,只触发了一次ADC转换。
2L大大,我试过你的方法了,还是不行,我在想是不是我的TIM时钟频率或者是ADC的采样时间间隔有问题,或者它们之间搭配出现问题了,请各位不吝指教。

我一直按调试下一步,让TIM器时间间隔到,但DR和我自己定义的内存,数据都没更新。
qq_19269621 2017-08-25
  • 打赏
  • 举报
回复
我之前也遇到了只触发一次的问题,后来在 ADC_DMACmd(ADC1, ENABLE); 之前加了一句 ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE); 解决的。
qq_19269621 2017-08-25
  • 打赏
  • 举报
回复
1.LZ,我看了一下之前的代码,连续转换是关闭的; 2.我当时的用法是 TIM定时触发 ADC进行采样,DMA将数据从 ADC的寄存器搬运到内存,缓冲区全满之后触发一个 DMA中断,中断函数里面再把数据拷贝出来进行一次处理; 3.之前我说的只触发一次是指,第一次缓冲区全满触发中断之后,DMA就不再搬运数据了,和LZ遇到的问题还不一样; 4.建议LZ参考标准库中的例程,多尝试修改,祝LZ早日解决问题;
  • 打赏
  • 举报
回复
现在有2个想法在我脑子了,有点困惑哦。 1.我这样写程序的初衷是为了靠TIM控制ADC采样,但不是用TIM的中断。例如我要一个周波内(20ms)采20个点,那配成每1.05ms,定时器计数溢出,触发一次单次ADC采样,从而达到目的。 2. 如果ADC采样配成连续模式,TIM触发一次后ADC自己就一直在转换,采样的时间间隔 = ADC转换通道的转换时间+ADC通道之间的间隔,其实这个时间很快的,也达不到我要的目的。 是不是要达到我这种目的不是这样配的。
tianxj001 2017-08-24
  • 打赏
  • 举报
回复
DMA不依赖CPU独立工作,也就是一旦你CPU进入while(1)它照样在后台操作,是不会打断你这个CPU死循环的。 初步看看你的程序,CAD也是通过TIM事件自动触发,也就是配置完成后,不会进入任何程度代码段,而由硬件自己自动工作,它转换完成后,再申请DMA,这也不需要CPU干涉,也就是你在程序流里面是看不到过程的,至于DMA是不是正常工作,你必须在调试时候观察DMA目标RAM内容是不是被DMA在及时更新,至于你这个测试程序,运行后进入while(1)是非常正常的。

27,375

社区成员

发帖
与我相关
我的任务
社区描述
硬件/嵌入开发 单片机/工控
社区管理员
  • 单片机/工控社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧