STM32之ADC_2(DMA实例)

发布者:koimqerolulk最新更新时间:2018-05-13 来源: eefocus关键字:STM32  ADC  DMA 手机看文章 扫描二维码
随时随地手机看文章

实例分析:(采用DMA模式)


mian函数: 

extern __IO u16 ADC_ConvertedValue;//ADC转换的电压值,是在ADC1_Init()所属的文本中定义的 

float ADC_ConvertedValueLocal;//用来保存转换计算后的电压值 

int main(void) 

USART1_Config(); //串口配置 

ADC1_Init(); //使能ADC1以及配置ADC1为DMA模式


while (1)

{

    ADC_ConvertedValueLocal =(float) ADC_ConvertedValue/4096*3.3; // 读取转换的 AD 值


    printf("\r\n The current AD value = 0x%04X \r\n", ADC_ConvertedValue); // ADC_ConvertedValue为全局变量,SRAM读取出来的ADC1转换的值


    printf("\r\n The current AD value = %f V \r\n",ADC_ConvertedValueLocal);//局部变量,用来保存转换计算后的电压值


    Delay(0xffffee); // 延时

}


}


main函数的功能就是向串口发送当前ADC1转换的电压值。(串口配置在前面的博客有说,这里就不分析了)


PS: 

ADC_ConvertedValue的值通过DMA获取的,但是在使用DMA时,由于不是内核执行的指令,所以修改变量值时不会出现赋值语句的。


ADC的初始化函数: 

void ADC1_Init() 

ADC1_GPIO_Config(); 

ADC1_Mode_Config(); 

ADC1_Init()调用了ADC1_GPIO_Config()和ADC1_Mode_Config().这两个函数的作用分别是配置好ADC1所用的I/O端口,配置ADC1初始化及DMA模式。


GPIO配置: 

static ADC1_GPIO_Config(void) 

GPIO_InitTypeDef GPIO_InitStructure;


RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA1的时钟


RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC,ENABLE); //使能GPIOC并且使能复用功能ADC1


GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN; //模拟输入

GPIO_Init(GPIOC,&GPIO_InitStructure);  //属于PC1,输入是不需要设置频率的


GPIO的配置很简单,首先是使能了DMA1,GPIOC,ADC1的时钟,然后把ADC1的通道11使用的GPIO引脚PC1配置成模拟输入模式.(在使用ADC的输入时,必须使用模拟输入,详细的可以查看之前的博客,GPIO工作方式)


PS: ADC1通道11 

每个ADC通道都对应着一个GPIO引脚端口,GPIO的引脚在设置为模拟输入模式后可用于模拟电压的输入端,STM32F103VET6是有三个ADC的,这三个ADC共用了16个外部通道。 

这里写图片描述 
(ADC通道引脚表,表中写的ADC12_INx x:表示4~9或者14~15,ADC12:表示可以使用ADC1_IN或者ADC2_IN.这里用到PC1在表中对应的是ADC123_IN11,所以可以使用ADC1的通道11,ADC2的通道11或者ADC3的通道11来采集PC1上模拟电压数据)

DMA以及ADC的配置 
static ADC1_Mode_Config(void) 

DMA_InitTypeDef DMA_InitStructure; 
ADC_InitTypeDef ADC_InitStructure; 
/这里是DMA的配置/ 
DMA_DeInit(DMA1_Channel1);

DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;//ADC1的地址    

DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;//内存地址  

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;//外设为数据源

DMA_InitStructure.DMA_BufferSize = 1;

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址固定

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;//内存地址固定

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //外设传输数据单位半字

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //内存传输数据单位半字

DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //循环传输

DMA_InitStructure.DMA_Priority = DMA_Priority_High;

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //关闭内存到内存的传输

DMA_Init(DMA1_Channel1, &DMA_InitStructure); //填充DMA,配置为DMA1的通道11


DMA_Cmd(DMA1_Channel1, ENABLE); //使能通道11


/*这里是ADC的配置*/

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//独立的ADC模式

ADC_InitStructure.ADC_ScanConvMode = DISABLE ;//关闭扫描模式(扫描一般用于多通道采集)

ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//开启连续转换模式(就是不停的进行ADC转换)

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //不使用外部触发转换

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//采集的数据右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1; //要转换的通道数量是1个

ADC_Init(ADC1, &ADC_InitStructure);//填充ADC1

RCC_ADCCLKConfig(RCC_PCLK2_Div8); //设置ADC时钟的分频,为PCLK2的8分频,就是9HZ 72/8 = 9

ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 1, ADC_SampleTime_55Cycles5);//配置ADC1的通道11为55.5个采样周期


/*这里是ADC的复位校准操作*/

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

ADC_Cmd(ADC1, ENABLE);   //使能ADC1


ADC_ResetCalibration(ADC1);//复位ADC1校准寄存器

while(ADC_GetResetCalibrationStatus(ADC1)); //等待校准寄存器复位成功


ADC_StartCalibration(ADC1);//ADC1校准

while(ADC_GetCalibrationStatus(ADC1));//等待ADC1校准成功


ADC_SoftwareStartConvCmd(ADC1, ENABLE);//使用软件触发ADC转换,因为没有采用外部触发


ADC的DMA配置:(详细的可以查看之前的博客STM32之DMA的配置代码分析) 
使用DMA1的通道1,数据从ADC外设的数据寄存器(ADC1_DR_Address)转移到内存(ADC_ConvertedValue 变量),内存、外设地址都固定,每次传输的数据大小为半字(16位),使用 DMA 循环传输模式。 
(其中ADC1外设的DMA请求通道为DMA1的通道1,初始化时要注意.) 
DMA传输的外设地址ADC1_DR_Address是一个自定义宏#define ADC1_DR_Address ((u32)0x40012400+0x4c),ADC_DR数据寄存器保存了ADC转换后的数值,以它作为DMA的传输源地址,它的地址是ADC1外设的基地址(0X4001 2400)加上ADC数据寄存器(ADC_DR)的地址偏移(0x4c)计算得到的. 
(ADC起始地址说明表以及ADC_DR寄存器描述可以在STM32参考手册找到) 
起始地址表大概如下: 
起始地址 外设 
0x4001 2800~0x4001 2BFF ADC2 
0x4001 2400~0x4001 27FF ADC1

ADC_DR寄存器描述及其地址偏移 
这里写图片描述

ADC配置: 
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; 
STM32具有多个ADC,而不同的ADC又有共用通道,当两个ADC采集一个通道的先后顺序,时间间隔不同,就会出现了各种各样的模式,例如:同步注入模式,同步规则模式等10种.(这里因为用于测量电阻分压后的电压值,所以要求不高,只使用一个ADC就可以满足要求了,因此ADC_Mode被赋值成ADC_Mode_Independent.(独立模式))

ADC_InitStructure.ADC_ScanConvMode = DISABLE; 
如果有多个通道需要采集信号,可以把ADC配置成按一定的顺序来对各个通道进行扫描转换(就是轮流采集各个通道的值).如果是多个通道采集,就必须开启这个模式.(因为这里只用了1个通道,所以禁止使用扫描模式)

ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; 
连续转换模式,ADC转换模式有两个,一个是连续转换一个是单次转换. 
单次转换:ADC只采集一次数据就停止. 
连续转换:上一次ADC转换完成后,立即开启下一次转换. 
这里需要不断采集电压值,所以使能连续转换模式.

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; 
ADC是需要接收到触发信号才开始进行模数转换,这些触发信号可以是外部中断触发(EXTI线),定时器触发.这两个都是外部触发信号,如果不使用外部触发信号可以使用软件控制触发(这里只用了软件控制触发,所以禁止外部触发)

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; 
数据的对齐方式.ADC在转换后的数值是被保存到数据寄存器(ADC_DR)的0~15位或者16~32位,数据的宽度为16位,而ADC转换精度为12位.所以把12位的数据保存到16位的区域,就涉及到数据对齐.(这里的左右对齐其实和word文档中的文本左,右对齐是一个意思). 
左对齐:ADC转换的数值最高位D12与存储区域的最高位Bit 15对齐,也就是说存储区域的低4位无意义. 
右对齐:ADC转换的数值最低位D0与存储区域的最低位Bit 0对齐,也就是说存储区域的高4位无意义. 
(这里右对齐比较方便,所以这里用了右对齐)

ADC_InitStructure.ADC_NbrOfChannel = 1; 
保存了进行ADC数据转换的通道数量,可以是1~16(一共16个通道),这里只采集了PC1这个通道,所以赋值为1.

RCC_ADCCLKConfig(RCC_PCLK2_Div8); 
填充完结构体后,就调用了外设初始化函数ADC_Init(ADC1, &ADC_InitStructure);进行初始化了.在使能DMA和ADC1时,还需要设置ADC的时钟(ADCCLK),ADC的频率越高,转换的速率也就越快,但是ADC时钟有上限值,不能超过14MHz. 
这里写图片描述

ADC的时钟(ADCCLK)为ADC预分频器的输出,而ADC预分频器的输入则为高速外设时钟(PCLK2).使用RCC_ADCCLKConfig()库函数实质就是设置ADC的预分频值,可以设置为PCLK2的2,4,6,8分频. 
PS: 
PCLK2的常用时钟频率为72MHz,而ADCCLK的频率必须低14MHz,所以ADCCLK最高频率为PCLK2的8分频(ADDCLK=9MHz),如果希望使ADC以最高频率14MHz运行,可以把PCLK2配置为56MHz,然后用4分频得到ADCCLK(14MHz).

ADC_RegularChannelConfig(ADC1,ADC_Channel_11,1,ADC_SampleTime_55Cycles5); 
设置ADC采样周期,ADC的转换时间不仅与ADC的时钟有关,还与采样周期有关.(每个不用的ADC通道,都可以配置为不同的采样周期) 
这个库函数的原形是: 
voidADC_RegularChannelConfig(ADCTypeDef*ADCx,uint8_t ADC_Channel,uint8_t Rank,uint8_t ADC_SampleTime) 
ADCx:选用的那个ADC 如ADC1,ADC2,ADC2

ADC_Channel:选择要配置的ADC通道(其中16,17是内部通道 16是连接至芯片的温度传感器,17是连接至内部电源模块)

Rank:配置为多通道扫描时,此通道的采样顺序,例如通道1,4,7的Rank值分别配置成3,2,1.那么ADC扫描时,扫描的顺序就是通道7->通道4->通道1

ADC_SampleTime:配置本通道的采样周期,最短可配置为1.5个采样周期(这里的周期是ADCCLK的时钟周期) 
这里面是把ADC1通道11配置为55.5个采样周期,因为前面ADCCLK在前面已经配置为9MHz,根据STM32的ADC采样时间计算公式: 
TCONV = (采样周期+12.5个周期)/ADCCLK 
所以这里的TCONV = (55.5+12.5)/9 大约等于7.56us.

ADC的自校准: 
在开始ADC转换之前,需要启动ADC的自校准.ADC有一个内置自校准模式(校准可以大幅减小因内部电容器组的变化而造成的准精度误差),在校准期间,在每个电容器上都会计算出一个误差修正码(数字值).这个码是用于消除在随后的转换中每个电容器上产生的误差. 
代码: 
ADC_ResetCalibration(ADC1);//复位ADC1校准寄存器 
while(ADC_GetResetCalibrationStatus(ADC1)); //等待校准寄存器复位成功

ADC_StartCalibration(ADC1);//ADC1校准
while(ADC_GetCalibrationStatus(ADC1));//等待ADC1校准成功123

调用了复位校准函数ADC_ResetCalibration()以及开始校准函数ADC_StartCalibration(),必须检查标志位等待校准完成,确保完成后才开始ADC转换.(建议是每次上电后都校准一次咯)

ADC_SoftwareStartConvCmd(ADC1, ENABLE); 
配置ADC1的模式为软件触发方式.

调用这个函数之后,ADC就开始进行转换了,每次转换完成后,由DMA控制器把转换从ADC数据寄存器(ADC_DR)中转移到变量ADC_ConvertedValue中,当DMA传输完成后,在main函数中使用 ADC_ConvertedValue的内容就是ADC的转换值了.

计算电压值: 
在main函数中,ADC_ConvertedValueLoca是一个float类型变量,它保存了有转换值计算出来的电压值,计算的公式是ADC通用的

实际电压 = ADC转换值*LSB 
LSB为Vref+接的参考电压/ADC的精度( LSB =3.3/2的12次方)

PS: 
这里面ADC_ConvertedValue是用volatile修饰的,用 volatile 声明的类型变量表示可以被某些编译器未知的因素更改,比如:操作系统、硬件或者其它线程等。因为 ADC_ConvertedValue 这个变量值随时都是会被 DMA 控制器改变的,所以用 volatile 来修饰它,确保每次读取到的都是实时的 ADC 转换值.


关键字:STM32  ADC  DMA 引用地址:STM32之ADC_2(DMA实例)

上一篇:基于STM32的12864串行时序的实现
下一篇:STM32 使用DMA处理ADC 学习笔记!!

推荐阅读最新更新时间:2024-03-16 16:02

采用开关电源为高速模数转换器供电
  系统设计工程师常被要求降低总体功耗,以减少对我们环境的影响,同时降低投资和运营成本。他们还需要提高电路密度,以便实现外形尺寸更小的电子系统,并且能在更严苛的环境下工作。遗憾的是,若将高功耗解决方案整合到这些系统中,会带来极大的散热问题,而使得其他目标也无法实现。   传统上,ADC制造商一般推荐采用线性稳压器为转换器提供干净的电源。线性稳压器能够抑制系统电源中经常出现的低频噪声。此外,铁氧体磁珠和去耦电容相结合的方法可用来减少高频噪声。这种方法虽然有效,但却限制了效率,特别是在线性稳压器必须从高出其输出电压几伏的电源轨进行降压调节的系统中。低压差稳压器(LDO)的效率通常为30%~50%,而DC/DC稳压器的效率则高达90
[电源管理]
采用开关电源为高速<font color='red'>模数转换器</font>供电
stm32单片机按键控制的用法解析
1 /* 2 ::按键控制 3 PA8接LED,PE2接按键 4 */ 5 #include“stm32f10x.h” 6 void RCC_Configuration(void); 7 void GPIO_Config(void); 8 void Delay(__IO uint32_t nCount); 9 10 int main() 11 { 12 RCC_Configuration(); //系统时钟配置|使能GPIO口 13 GPIO_Config(); //LED控制配置 14 while (1) 15 { 16 if(!GPIO_ReadInputDataBit(GPIOE,GPIO_Pin_2)) 17 { 18
[单片机]
STM32外部中断总结
该文章是学习了STM32后所写,是对STM32的小小的理解。在文中若有错误指出请指正,不胜感激。 STM32有最多68个可屏蔽的中断通道,有16个可编程的优先级。对于STM32F103ZE芯片,其内部包含了19个边沿检测器用来产生中断或者事件请求。对于外部中断只用了16个外部触发输入线。每一个外部输入线都能够被独立的屏蔽,pending register的寄存器能够保存输入线的状态,这个寄存器不能通过硬件自动清零,必须使用软件来清零; 外部中断的内部连接图如下: 触发信号通过输入线进入检测器,检测器的功能是可以设置的,就是通过上边两个边沿检测器即上升沿出发选择寄存器和下降沿出发选择寄存器。从图中就可以看出外部中断/事件触发
[单片机]
stm32与AT24C02的I2C通信总结(模拟时序)
从51的时候就学习了I2C通信协议,但51的功能就那些,内部没有集成I2C模块,所以只能通过模拟I2C通信的时序来和EEPROM进行通信,stm32内部集成了I2C通信的片上外设,但由于内部I2C外设复杂和不稳定行,所以用的人不是很多,而基本上使用I2C的通信都是通过模拟时序的方式来实现的 首先I2C是同步半双工的通信方式,需要两条线即可,SCL时钟线,同步时钟由主机产生,SDA数据线用来发送接收数据,任何时候只能一台主机发送数据。 在编程的时候碰到了很多问题,其中一项就是等待从机的应答程序 开始的时候编写如下 void iic_wait_ack() 错误 voi
[单片机]
<font color='red'>stm32</font>与AT24C02的I2C通信总结(模拟时序)
STM32双核板的应用设计与ISP的从机软件升级
引言 在单片机的应用设计中,常常会遇到如下问题:其一,某一熟悉类型的单片机功能可用,性价比也很好,但限于某种内部资源(如串口数、A/D路数等)不足,不得不选用更高档或不太熟悉的单片机,造成资源的浪费和开发周期的延长。其二,在海洋远程监测等重要领域,对控制器的可靠性要求较高,而单片机存在死机的可能性,即使可以通过配备看门狗来避免这种情况,但这种 粗暴 的复位方式并不合理(首先,复位打乱了正常的数据采集和处理工作,导致重要数据丢失;其次,即便能记录下复位时间和次数,但复位原因和复位前状态等信息无法侦测,一些本该解决的BUG被掩盖,导致频繁复位)。其三,由于开发周期不足或测试不充分,导致设备投入运行后出现故障,而这些故障往往通过软件升级的
[单片机]
<font color='red'>STM32</font>双核板的应用设计与ISP的从机软件升级
STM32学习笔记之GPIO的基本使用
楼学习中主要使用的资料有 战舰V3资料盘——教学视频 《STM32F1开发指南-库函数版本_V3.1》 (PDF) 《零死角玩转STM32》 野火出版 使用的学习板为屹讯电子嵌入式平台v2(MCU为 STM32F103RCT6) **某些IO口带有FT标识,代表其最大工作电压达到5V CPU能直接读取外设电平 上拉电阻的作用????????????????? 下拉电阻 CPU能直接读取外部电压变化(模拟量) 0-3.3v 输出控制电路置1,输出由外部电路决定(上拉或下拉) 输出控制电路置0,输出为0 置1置0的写入由CPU执行 输出控制电路的写入由外设模块决定 输出控制电路置1,输出1;置0,输出0。
[单片机]
<font color='red'>STM32</font>学习笔记之GPIO的基本使用
CORTEX-M3与STM32_M3内核STM32的三相多功能电能表解决方案
电能表作为电能计量的基本设备,受到国家电力部门的长期重视,电能表生产企业更是不遗余力地寻求设计与开发性能俱佳且成本更低的解决方案。目前国内的电能表设计已经走过了由8位MCU向通用DSP甚至专用DSP的变革,通用DSP的应用方案的劣势在于DSP的专业应用和嵌入程度不够深的问题,成本偏高;而专用 DSP功能相对固定,这样给电能表设计和生产厂家带来功能差异化空间不足的困难。基于ARM的方案也已经出现,但是适合应用的ARM7 TDMI在性能上不尽人意,同时外设资源不足;而更高端的ARM9系统的复杂程度很高,成本也较高。选择一颗合适且低成本的微处理器日益成为电能表行业的关键所在,直到意法半导体公司(STMicroelectronic公司,下
[单片机]
CORTEX-M3与STM32_M3内核<font color='red'>STM32</font>的三相多功能电能表解决方案
STM32外部中断解决方法
01 单片机外部中断简介 所谓外部中断,就是通过外部信号所引起的中断,如单片机引脚上的电平变化(高电平、低电平)、边沿变化(上升沿、下降沿)等。51单片机有5个中断源,其中有两个是外部中断,分别为INT0和INT1,INT0被分配在P3.2引脚,INT1被分配在P3.2引脚,也即是说如果使用51单片机的外部中断0,则必须将信号接在P3.2上,否则无效。 02 举例说明什么是中断 单片机在执行程序时有两种方式: 查询方式 中断方式 所谓查询方式就是单片机一遍一遍的扫描,查看所监视的目标有没有发生变化,是一种主动式的监视方法,用一个成语可以很客观的描述:守株待兔。 所谓中断方式就是单片机不主动去监视目标,而是目标主动通知单片机状态
[单片机]
<font color='red'>STM32</font>外部中断解决方法
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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