Stm32的ADC有DMA功能这都毋庸置疑,也是我们用的最多的!然而,如果我们要对一个信号(比如脉搏信号)进行定时采样(也就是隔一段时间,比如说2ms),有三种方法:

1、使用定时器中断每隔一定时间进行ADC转换,这样每次都必须读ADC的数据寄存器,非常浪费时间!

2、把ADC设置成连续转换模式,同时对应的DMA通道开启循环模式,这样ADC就一直在进行数据采集然后通过DMA把数据搬运至内存。但是这样做的话还得加一个定时中断,用来定时读取内存中的数据!

3、使用ADC的定时器触发ADC转换的功能,然后使用DMA进行数据的搬运!这样只要设置好定时器的触发间隔,就能实现ADC定时采样转换的功能,然后可以在程序的死循环中一直检测DMA转换完成标志,然后进行数据的读取,或者使能DMA转换完成中断,这样每次转换完成就会产生中断,我是采用第二种方法。下面上代码:我这里使用的单通道

//定时器初始化

void TIM2_Configuration(void)

 TIM_TimeBaseInitTypeDef
TIM_TimeBaseStructure; 
 TIM_OCInitTypeDef
TIM_OCInitStructure; 
 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2 ,
ENABLE); 
 TIM_TimeBaseStructure.TIM_Period =
1999;//设置2ms一次TIM2比较的周期
 TIM_TimeBaseStructure.TIM_Prescaler =
71;//系统主频72M,这里分频71,相当于1000K的定时器2时钟 
 TIM_TimeBaseStructure.TIM_ClockDivision =
0x0; 
 TIM_TimeBaseStructure.TIM_CounterMode =
TIM_CounterMode_Up;
 TIM_TimeBaseInit(TIM2, &
TIM_TimeBaseStructure);

TIM_OCInitStructure.TIM_OCMode =
TIM_OCMode_PWM1;//下面详细说明 
 TIM_OCInitStructure.TIM_OutputState =
TIM_OutputState_Enable;//TIM_OutputState_Disable;

TIM_OCInitStructure.TIM_Pulse =
1000; 
 TIM_OCInitStructure.TIM_OCPolarity =
TIM_OCPolarity_Low;//如果是PWM1要为Low,PWM2则为High

TIM_OC2Init(TIM2, &
TIM_OCInitStructure);

// TIM_InternalClockConfig(TIM2);
// TIM_OC2PreloadConfig(TIM2,
TIM_OCPreload_Enable); 
// TIM_UpdateDisableConfig(TIM2, DISABLE);
}

//ADC_DMA初始化配置

void ADC_DMA_Config(void)
{
  DMA_InitTypeDef DMA_InitStructure; //
注:ADC为12位模数转换器,只有ADCConvertedValue的低12位有效
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,
ENABLE);//使能DMA时钟
 
DMA_DeInit(DMA1_Channel1);//开启DMA1的第一通道 
  DMA_InitStructure.DMA_PeripheralBaseAddr =
ADC1_DR_Address;//DMA对应的外设基地址
  DMA_InitStructure.DMA_MemoryBaseAddr =
(uint32_t)&ADCConvertedValue; //内存存储基地址
  DMA_InitStructure.DMA_DIR =
DMA_DIR_PeripheralSRC; //DMA的转换模式为SRC模式,由外设搬移到内存
  DMA_InitStructure.DMA_BufferSize =
1;//DMA缓存大小,1个
  DMA_InitStructure.DMA_PeripheralInc =
DMA_PeripheralInc_Disable; //接收一次数据后,设备地址禁止后移

DMA_InitStructure.DMA_MemoryInc =
DMA_MemoryInc_Disable; //关闭接收一次数据后,目标内存地址后移
  DMA_InitStructure.DMA_PeripheralDataSize =
DMA_PeripheralDataSize_HalfWord;//定义外设数据宽度为16位
  DMA_InitStructure.DMA_MemoryDataSize =
DMA_MemoryDataSize_HalfWord; //DMA搬移数据尺寸,HalfWord就是为16位
  DMA_InitStructure.DMA_Mode
=DMA_Mode_Circular;//循环转换模式
  DMA_InitStructure.DMA_Priority =
DMA_Priority_High;//DMA优先级高
  DMA_InitStructure.DMA_M2M =
DMA_M2M_Disable;//M2M模式禁用
  DMA_Init(DMA1_Channel1,
&DMA_InitStructure);

DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,
ENABLE);//使能传输完成中断
}

//ADC初始化

void PulseSenosrInit(void)
{
//当外部触发信号被选为ADC规则或注入转换时,只有它的上升沿可以启动转换

ADC_InitTypeDef ADC_InitStructure;
  ADC_GPIO_Configuration();//IO口配置
 
TIM2_Configuration(); //定时器配置
  ADC_DMA_Config();//ADC_DMA配置
  ADC_InitStructure.ADC_Mode =
ADC_Mode_Independent; //独立的转换模式 ADC_DUALMOD[3:0]=0000;
  ADC_InitStructure.ADC_ScanConvMode
=DISABLE;//关闭扫描模式 因为只有一个通道
  ADC_InitStructure.ADC_ContinuousConvMode
=DISABLE;//关闭连续转换模式 否则只要触发一次,
 
//后续的转换就会永不停歇(除非CONT清0),这样第一次以后的ADC,就不是由TIM2_CC2来触发了
  ADC_InitStructure.ADC_ExternalTrigConv
=ADC_ExternalTrigConv_T2_CC2;//软件转换模式
  ADC_InitStructure.ADC_DataAlign =
ADC_DataAlign_Right;//对齐方式,ADC为12位中,右对齐方式 ADC_ALIGN=0;
  ADC_InitStructure.ADC_NbrOfChannel =
1;//开启通道数,1个  ADC_SQR1[23:20]=0000;
  //ADC_SQR1[23:20] 设置通道数目的选择
  ADC_Init(ADC1, &ADC_InitStructure);
 //
RCC_ADCCLKConfig(RCC_PCLK2_Div6);//配置时钟(12MHz),在RCC里面还应配置APB2=AHB时钟72MHz

ADC_RegularChannelConfig(ADC1, ADC_Channel_2,
1,ADC_SampleTime_1Cycles5);
  //ADC_SMPR2 ADC_SMPR1
设置每个通道的采样时间 
 
//ADC_SQR1[19:0]DC_SQR1[29:0]DC_SQR3[29:0] 
设置对应通道的转换顺序  适用于多通道采样
  //ADC通道组, 第3个通道 采样顺序1,转换时间
  ADC_ExternalTrigConvCmd(ADC1,
ENABLE);//设置外部触发模式使能(这个“外部“其实仅仅是相//对于ADC模块的外部,
 
  ADC_DMACmd(ADC1,
ENABLE);

ADC_Cmd(ADC1, ENABLE); 
//ADC命令,使能  ADC_ADON=1
    
 
ADC_ResetCalibration(ADC1);  
//重新校准
 
 
while(ADC_GetResetCalibrationStatus(ADC1)); 
//等待重新校准完成
 
 
ADC_StartCalibration(ADC1);  //开始校准 
ADC_RSTCAL=1; 初始化校准寄存器
 
 
while(ADC_GetCalibrationStatus(ADC1));   
//等待校准完成 
ADC_CAL=0;  
 
  
//ADC_SoftwareStartConvCmd(ADC1, ENABLE);
//连续转换开始,ADC通过DMA方式不断的更新RAM区。
  //ADC_SWSTART=1 开始规则转换 切记
软件触发也属于外部事件  要设置 
ADC_EXTTRIG=1
////  //实际上还是在STM32内部)
  TIM_Cmd(TIM2, ENABLE);//最后面打开定时器使能
  DMA_Cmd(DMA1_Channel1,
ENABLE);//使能DMA

}

//中断处理函数

void  DMA1_Channel1_IRQHandler(void)
{
 
 if(DMA_GetITStatus(DMA1_IT_TC1)!=RESET){

//自己的中断处理代码
但是记住程序不要太复杂  最好不要超过中断时间
        
DMA_ClearITPendingBit(DMA1_IT_TC1);
 }
 }

//中断配置

NVIC_InitTypeDef
NVIC_InitStructure;  
   
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
   
NVIC_InitStructure.NVIC_IRQChannel
=DMA1_Channel1_IRQn;  
   
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
   
NVIC_InitStructure.NVIC_IRQChannelSubPriority =
0; 
   
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
   
NVIC_Init(&NVIC_InitStructure);

void
ADC_GPIO_Configuration(void)       
//ADC配置函数
{
  GPIO_InitTypeDef GPIO_InitStructure;
 
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1|RCC_APB2Periph_GPIOA,
ENABLE);  
//使能ADC和GPIOA时钟

GPIO_InitStructure.GPIO_Pin =
GPIO_Pin_2;       
//管脚2
  GPIO_InitStructure.GPIO_Mode =
GPIO_Mode_AIN;   
//模拟输入模式
  GPIO_Init(GPIOA,
&GPIO_InitStructure);    
//GPIO组
}

关于Stm32定时器+ADC+DMA进行AD采样的实现的更多相关文章

  1. STM32 F4 ADC DMA Temperature Sensor

    STM32 F4 ADC DMA Temperature Sensor Goal: detecting temperature variations using a temperature senso ...

  2. stm32的ADC规则组通道采样顺序设置

    先看一下固件库手册 再看一下手册上的例子:  有两个通道,,并且顺序如下

  3. STM32定时器触发ADC多通道连续采样,DMA缓存结果

    STM32的ADC使用非常灵活,采样触发方面:既支持软件触发,定时器或其他硬件电路自动触发,也支持转换完成后自动触发下一通道/轮转换.转换结果存储方面:既支持软件读取和转存,也支持DMA自动存储转换结 ...

  4. STM32 多通道ADC采样,采用Timer1进行采样率控制,利用DMA进行传输

    http://blog.csdn.net/varding/article/details/17559399 http://www.51hei.com/stm32/3842.html https://w ...

  5. STM32 双ADC同步规则采样

      最近需要用到两个ADC对电压电流进行同步采样,看了一下STM32的ADC介绍,发现STM32最多有3个独立ADC,有在双AD模式下可以进行同步测量,正好满足我的要求.参考官方给的例子在结合自己的需 ...

  6. STM32之ADC实例(基于DMA方式)

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/zouleideboke/article/details/75112224 ADC简介: ADC(An ...

  7. STM32中AD采样的三种方法分析

    在进行STM32F中AD采样的学习中,我们知道AD采样的方法有多种,按照逻辑程序处理有三种方式,一种是查询模式,一种是中断处理模式,一种是DMA模式.三种方法按照处理复杂方法DMA模式处理模式效率最高 ...

  8. 【STM32】用DMA实现多路ADC通道数据采集

    今天尝试了下STM32的ADC采样,并利用DMA实现采样数据的直接搬运存储,这样就不用CPU去参与操作了. 找了不少例子参考,ADC和DMA的设置了解了个大概,并直接利用开发板来做一些实验来验证相关的 ...

  9. 【STM32H7教程】第46章 STM32H7的ADC应用之DMA方式多通道采样

    完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980 第46章       STM32H7的ADC应用之DMA方式多 ...

随机推荐

  1. python常用模块之时间模块

    python常用模块之时间模块 python全栈开发时间模块 上次的博客link:http://futuretechx.com/python-collections/ 接着上次的继续学习: 时间模块 ...

  2. 开启mysql的远程访问权限

    改表法 1.登陆mysql mysql -u root -p 2.修改mysql库的user表,将host项,从localhost改为%.%这里表示的是允许任意host访问,如果只允许某一个ip访问, ...

  3. IdentityServer4-参考

    一.Identity Resource 二.API Resource 三.Client 四.GrantValidationResult 五.Profile Service 六.IdentityServ ...

  4. Lakeshore 中文开发界面,示例项目,飞机大战 等 Lakeshore Chinese development interface, sample project, aircraft war, etc

    Lakeshore 中文开发界面,示例项目,飞机大战 等 Lakeshore Chinese development interface, sample project, aircraft war, ...

  5. IE6条件下的bug与常见的bug及其解决方法

    1.IE6条件下有双倍的margin 解决办法:给这个浮动元素增加display:inline属性 2. 图片底部有3像素问题 解决办法:display:block;或者vertical-align: ...

  6. tcp nonblock connection rst

    客户端(>5w)异步connect连接到server端,server端listen backlog设置为1024,发现存在部分客户端建立连接后,收到服务端的rst包. 先看下tcp监听套接字维护 ...

  7. 大文件拆分方案的java实践(附源码)

    引子 大文件拆分问题涉及到io处理.并发编程.生产者/消费者模式的理解,是一个很好的综合应用场景,为此,花点时间做一些实践,对相关的知识做一次梳理和集成,总结一些共性的处理方案和思路,以供后续工作中借 ...

  8. ERROR: While executing gem … (Gem::RemoteFetcher::FetchError)

    原文地址:https://www.zfanw.com/blog/error-while-executing-gem-gem-remote-fetch-error.html 我对命令行下安装 gem 包 ...

  9. Scala:Next Steps in Scala

    Array val greetStrings = new Array[String](3) greetStrings(0) = "Hello" greetStrings(1) = ...

  10. 线程安全的CopyOnWriteArrayList介绍

    证明CopyOnWriteArrayList是线程安全的 先写一段代码证明CopyOnWriteArrayList确实是线程安全的. ReadThread.java import java.util. ...