历史上的今天
返回首页

历史上的今天

今天是:2025年03月12日(星期三)

正在发生

2019年03月12日 | 【STM32】ADC库函数、一般步骤详解 实例:内部温度传感器

2019-03-12 来源:eefocus

STM32F1xx官方资料:

《STM32中文参考手册V10》-第11章 模拟/数字转换(ADC)


《STM32中文参考手册V10》-第11章 第11.10小节 温度传感器


ADC采样数值

如何STM32的ADC模块,得到接入ADC管脚上的实际电压值?


会读到什么值

由于STM32的ADC是12位逐次逼近型的模拟数字转换器,也就是说ADC模块读到的数据是12位的数据。


因此:STM32读到的ADC值,是从0到4095(111111111111)。当把ADC引脚接了GND,读到的就是0;当把ADC引脚接了VDD,读到的就是4095。


读到的值怎么换算成实际的电压值

前面提到了,我们输入GND,读到的值是0,输入VDD,得到的值是4095,那么,当读到2035的时候,怎么求输入电压多少V吗?这个问题,归根接地,就到了数学XY坐标,已知两点坐标值(0,0)(3.3,4095),给出任意X坐标值,求Y值的问题了吧?简单不简单?



参考电压是什么

讨论这个问题之前,先拿万用表量一下你的VDDA的实际电压是多大?是不是标准的3.300V?应该不是吧?或许是2.296V,或许是3.312V。然后你把VDD连接到ADC引脚之后,得到的是4095;也就是,实际上,当你读出4095这个数据的时候,实际的电压值不是你想象中的3.300V。有些初学者,觉得几毫伏的电压差无所谓,但实际应用中,几毫伏就可能代表很大的实际工况,例如,在一个量程为50克的电子称上。


所以,这时候,芯片厂商就想了一个办法,给ADC模块中引入参考电压,由非常标准的参考电压芯片来接入参考电压引脚。标准的电压芯片,我们一般叫做参考电压芯片,或者叫做基准电压芯片。例如REF3133(输出3.300V)、REF3025(输出2.500V)等等。


ADC引脚的输入电压范围是多大

一般情况下,ADC引脚的输入电压,是从0~VDD,如果有REF引脚,一般是0~Vref,也有0~2Vref的情况。


如果被测的电压大于ADC的输入电压,例如,要用STM32测量0~5V的电压的话,可以在输入ADC引脚之前,加入电阻分压和放大器电路。


ADC相关配置库函数

1个初始化函数

void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);

作用:配置ADC模式、扫描模式、单次连续模式、外部触发方式、对齐方式、规则序列长度。


1个使能函数

void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);

作用:配置ADC使能。


1个软件转换函数

void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);

作用:ADC使能软件转换(在ADC_Init函数中,外部触发方式选择none)。


1个规则通道配置函数

void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime);

作用:配置某个ADC控制器的某个通道以某种采样率置于规则组的某一位(对应函数的四个参数:ADC控制器名、ADC通道名、规则组的第n个、采样率)。


1个获取转换结果函数

uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx);

作用:获得某个ADC控制器的软件转换结果。


ADC一般步骤

实例要求:利用ADC1的通道1(PA1)采集外部电压值。


开启PA口时钟和ADC1时钟,设置PA1为模拟输入。调用函数:GPIO_Init();APB2PeriphClockCmd();

复位ADC1,同时设置ADC1分频因子。调用函数:ADC_DeInit(ADC1);RCC_ADCCLKConfig(RCC_PCLK2_Div6);

初始化ADC1参数,设置ADC1的工作模式以及规则序列的相关信息。调用函数:void ADC_Init();

使能ADC并校准。调用函数:ADC_Cmd();

配置规则通道参数。调用函数:ADC_RegularChannelConfig();

开启软件转换:ADC_SoftwareStartConvCmd(ADC1);

等待转换完成,读取ADC值。调用函数:ADC_GetConversionValue(ADC1)。

//初始化ADC

//这里我们仅以规则通道为例

//我们默认将开启通道0~3    

void  Adc_Init(void)

{

ADC_InitTypeDef ADC_InitStructure; 

GPIO_InitTypeDef GPIO_InitStructure;

 

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE );   //使能ADC1通道时钟

 

RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

 

//PA1 作为模拟通道输入引脚                         

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚

GPIO_Init(GPIOA, &GPIO_InitStructure);

 

ADC_DeInit(ADC1);  //复位ADC1 

 

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式

ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模数转换工作在单通道模式

ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模数转换工作在单次转换模式

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目

ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   

  

ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1


ADC_ResetCalibration(ADC1); //使能复位校准  

 

while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束


ADC_StartCalibration(ADC1); //开启AD校准

 

while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束

 

}   

//获得ADC值

//ch:通道值 0~3

u16 Get_Adc(u8 ch)   

{

  //设置指定ADC的规则组通道,一个序列,采样时间

ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期       

  

ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能

 

while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

 

return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果

}

 

u16 Get_Adc_Average(u8 ch,u8 times)

{

u32 temp_val=0;

u8 t;

for(t=0;t

{

temp_val+=Get_Adc(ch);

delay_ms(5);

}

return temp_val/times;

}  

 int main(void)

 {  

  u16 adcx;

float temp;

delay_init();     //延时函数初始化   

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级

uart_init(115200); //串口初始化为115200

  LED_Init();      //LED端口初始化

LCD_Init();

  Adc_Init();   //ADC初始化

 

POINT_COLOR=RED;//设置字体为红色 

LCD_ShowString(60,50,200,16,16,"WarShip STM32");

LCD_ShowString(60,70,200,16,16,"ADC TEST");

LCD_ShowString(60,90,200,16,16,"ATOM@ALIENTEK");

LCD_ShowString(60,110,200,16,16,"2015/1/14");

//显示提示信息

POINT_COLOR=BLUE;//设置字体为蓝色

LCD_ShowString(60,130,200,16,16,"ADC_CH0_VAL:");       

LCD_ShowString(60,150,200,16,16,"ADC_CH0_VOL:0.000V");        

while(1)

{

adcx=Get_Adc_Average(ADC_Channel_1,10);

LCD_ShowxNum(156,130,adcx,4,16,0);//显示ADC的值

temp=(float)adcx*(3.3/4096);

adcx=temp;

LCD_ShowxNum(156,150,adcx,1,16,0);//显示电压值

temp-=adcx;

temp*=1000;

LCD_ShowxNum(172,150,temp,3,16,0X80);

LED0=!LED0;

delay_ms(250);

}

 }

STM32控制程序分析

Adc_Init()函数:ADC初始化函数。


这里需要注意的点有:ADC的时钟配置函数包括两步,不要遗漏ADC的分频:


RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE );   //使能ADC1通道时钟

RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

PA1的GPIO模式应配置成模拟输入:


GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚

建议在上电时执行一次ADC校准(这一部分的代码就是这个样子,直接照抄就行了):


ADC_ResetCalibration(ADC1); //使能复位校准  

while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束

ADC_StartCalibration(ADC1); //开启AD校准

while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束

Get_Adc()函数:获得ADC转换后的值。


这里需要注意的是,当使用ADC_SoftwareStartConvCmd()使能软件转换之后,需要使用ADC_GetFlagStatus()函数等待转换结束后,再进行ADC_GetConversionValue()输出。


        ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期       

ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能

while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果

main()函数:主函数,实现函数的调用(采集外部电压值)。


ADC采样得到的数字怎么转化为电压值呢?这里采用这个方法来计算。原理见本文最开头的部分讲解。


temp=(float)adcx*(3.3/4096);

 


STM32内部温度传感器

STM32有一个内部的温度传感器,可以用来测量CPU及周围的温度(TA)。该温度传感器在内部和ADCx_IN16输入通道相连接,此通道把传感器输出的电压转换成数字值。


内部温度传感器的特征

温度传感器模拟输入推荐采样时间是17.1μs;

STM32的内部温度传感器支持的温度范围为:-40~125度;

精度比较差,为±1.5℃左右。

基于这样的特性,因此:内部温度传感器更适合于检测温度的变化,而不是测量绝对温度。如果需要测量绝度温度,应该使用一个外部温度传感器。


内部温度传感器的框图


内部温度传感器的使用

要使用STM32的内部温度传感器,必须先激活ADC的内部通道,这里通过ADC_CR2的TSVREFE位(bit23)设置。设置该位为1则启用内部温度传感器。

内部温度传感器的温度计算

STM32的内部温度传感器固定的连接在ADC的通道16上,所以,在设置好ADC之后只要读取通道16的值,就是温度传感器返回来的电压值了。然后根据这个电压值,我们就可以计算出当前温度。


计算公式如下:


T(℃)={(V25-Vsense)/Avg_Slope}+25


上式中:V25=Vsense在25度时的数值(典型值为:1.43),Avg_Slope=温度与Vsense曲线的平均斜率(单位为mv/℃或uv/℃)(典型值为4.3Mv/℃)。


利用以上公式,我们就可以方便的计算出当前温度传感器的温度了。


 


内部传感器一般步骤

选择ADC_IN16输入通道,设置采样时间大于17.1us,开启ADC;

设置ADC_CR2的TSVREFE位,打开内部温度传感器;

读取ADC结果,并计算温度。

//初始化ADC

//这里我们仅以规则通道为例

//我们默认将开启通道0~3

void T_Adc_Init(void)  //ADC通道初始化

{

ADC_InitTypeDef ADC_InitStructure; 

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE );   //使能GPIOA,ADC1通道时钟

  

RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //分频因子6时钟为72M/6=12MHz

 

        ADC_DeInit(ADC1);  //将外设 ADC1 的全部寄存器重设为缺省值

 

ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式

ADC_InitStructure.ADC_ScanConvMode = DISABLE; //模数转换工作在单通道模式

ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //模数转换工作在单次转换模式

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐

ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目

ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器

 

ADC_TempSensorVrefintCmd(ENABLE); //开启内部温度传感器

 

ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1

 

ADC_ResetCalibration(ADC1); //重置指定的ADC1的复位寄存器

 

       while(ADC_GetResetCalibrationStatus(ADC1)); //获取ADC1重置校准寄存器的状态,设置状态则等待

 

ADC_StartCalibration(ADC1); //

 

while(ADC_GetCalibrationStatus(ADC1)); //获取指定ADC1的校准程序,设置状态则等待

}

u16 T_Get_Adc(u8 ch)   

{

 

ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道3,第一个转换,采样时间为239.5周期       

 

ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能

while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果

}

 

//得到ADC采样内部温度传感器的值

//取10次,然后平均

u16 T_Get_Temp(void)

{

u16 temp_val=0;

u8 t;

for(t=0;t<10;t++)

{

temp_val+=T_Get_Adc(ADC_Channel_16);   //TampSensor

delay_ms(5);

}

return temp_val/10;

}

 

 //获取通道ch的转换值

//取times次,然后平均

u16 T_Get_Adc_Average(u8 ch,u8 times)

{

u32 temp_val=0;

u8 t;

for(t=0;t

{

temp_val+=T_Get_Adc(ch);

delay_ms(5);

}

return temp_val/times;

}    

 

//得到温度值

//返回值:温度值(扩大了100倍,单位:℃.)

short Get_Temprate(void) //获取内部温度传感器温度值

{

u32 adcx;

short result;

  double temperate;

adcx=T_Get_Adc_Average(ADC_Channel_16,20); //读取通道16,20次取平均

temperate=(float)adcx*(3.3/4096); //电压值 

temperate=(1.43-temperate)/0.0043+25; //转换为温度值  

result=temperate*=100; //扩大100倍.

return result;

}

 int main(void)

 {  

short temp; 

delay_init();     //延时函数初始化   

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级

uart_init(115200); //串口初始化为115200

LED_Init();   //初始化与LED连接的硬件接口

  LCD_Init(); //初始化LCD

  T_Adc_Init();   //ADC初始化     

POINT_COLOR=RED;//设置字体为红色 

LCD_ShowString(30,50,200,16,16,"WarShip STM32");

LCD_ShowString(30,70,200,16,16,"Temperature TEST");

LCD_ShowString(30,90,200,16,16,"ATOM@ALIENTEK");

LCD_ShowString(30,110,200,16,16,"2015/1/14");   

POINT_COLOR=BLUE;//设置字体为蓝色      

LCD_ShowString(30,140,200,16,16,"TEMPERATE: 00.00C");       

while(1)

{

temp=Get_Temprate(); //得到温度值 

if(temp<0)

{

temp=-temp;

LCD_ShowString(30+10*8,140,16,16,16,"-"); //显示负号

}else LCD_ShowString(30+10*8,140,16,16,16," "); //无符号

LCD_ShowxNum(30+11*8,140,temp/100,2,16,0); //显示整数部分

LCD_ShowxNum(30+14*8,140,temp%100,2,16, 0X80); //显示小数部分

LED0=!LED0;

delay_ms(250);

}

这里和上一个的区别,主要就是需要开启内部温度传感器:


ADC_TempSensorVrefintCmd(ENABLE); //开启内部温度传感器

同时,添加了对计算出来的电压值的再计算,换算成温度,就不多赘述了。

推荐阅读

史海拾趣

ALD [Advanced Linear Devices]公司的发展小趣事

ALD(Advanced Linear Devices)是一家专注于模拟半导体器件和电源解决方案的公司。以下是关于ALD公司发展的5个相关故事:

  1. 公司成立与创始人: ALD公司成立于1985年,总部位于美国加州的米尔皮塔斯市。该公司由俄罗斯裔工程师及企业家Vladimir Larkhov先生创立。Larkhov先生在半导体领域有着丰富的经验和深厚的技术底蕴,他创立ALD旨在为市场提供高品质、高性能的模拟集成电路和解决方案。

  2. 技术创新与专利: ALD公司以其在模拟半导体领域的技术创新而闻名。该公司拥有众多的专利,涵盖了多个关键技术领域,包括功率管理、电源管理、精密电压参考和模拟信号处理等。这些专利为ALD提供了技术壁垒,保护了其在市场上的地位。

  3. 产品线与市场应用: ALD公司的产品涵盖了模拟集成电路(ICs)、功率管理芯片、电源解决方案等领域。其产品被广泛应用于工业控制、医疗设备、汽车电子、通信设备、消费类电子产品等多个市场领域。ALD产品的高性能、高可靠性得到了客户的认可和信赖。

  4. 全球业务拓展: 随着全球市场的不断扩大,ALD公司不断加大对海外市场的开拓力度。除了在美国设立总部和研发中心外,ALD还在亚洲、欧洲等地设立了销售办事处和合作伙伴,以便更好地服务全球客户。

  5. 不断创新与发展: ALD公司一直致力于不断创新和发展,不断推出具有竞争力的新产品和解决方案,以满足客户不断变化的需求。公司注重技术研发和人才培养,拥有一支技术过硬、富有创新精神的团队,为公司的持续发展提供了强有力的支持。

这些故事展现了ALD公司在技术创新、产品应用、全球业务拓展和持续发展等方面的发展历程和成就。

ETAL公司的发展小趣事

ETAL公司成立于XXXX年,由一群富有远见和热情的电子工程师创立。他们看到了电子技术在全球范围内的广泛应用和巨大潜力,决定投身于这一行业。起初,ETAL主要专注于电子元器件的研发和生产,通过不断的技术创新和产品优化,逐渐在市场上树立了良好的口碑。

方舟(ARKLED)公司的发展小趣事

方舟公司不仅关注经济效益,还积极履行社会责任。公司积极参与环保事业,推动绿色生产,减少能源消耗和废弃物排放。同时,方舟公司还关注社会公益事业,捐资助学、扶贫济困,为社会和谐发展贡献了一份力量。

长江微电(cjiang)公司的发展小趣事

为了进一步扩大市场份额和品牌影响力,长江微电积极寻求与业内知名分销商的合作。基于对世强先进O2O技术分销模式的信赖与认可,长江微电与世强先进(深圳)科技股份有限公司签署了授权代理协议。双方将携手合作,共同为用户提供车规级电感、车规级耦合电感等全线产品。这一合作不仅丰富了世强硬创平台电子元器件产品的品类及信息,也为长江微电的市场拓展提供了有力支持。

BULGIN公司的发展小趣事

作为公认的环境密封连接器及元器件领先制造商,BULGIN公司一直专注于提升在环境密封、电源及防控产品领域的市场地位。公司凭借在极具挑战性的环境中提供可靠、坚固的电源、信号和数据连接的能力,赢得了客户的广泛赞誉。其中,备受欢迎的Buccaneer圆形连接器系列在业界享有出色的声誉,成为公司的一大亮点。

这五个故事展示了BULGIN公司在电子行业中的发展历程和关键里程碑。从创立初期到成为全球领先的电子零件制造商,再到战略性合并和业务扩展,BULGIN始终保持着创新精神和对优质产品的追求。如今,作为环境密封连接器及元器件的领先制造商,BULGIN将继续为客户提供卓越的产品和服务,推动电子行业的持续发展。

谷峰(GOFORD)公司的发展小趣事

GStek登丰微电子成立于2007年,正值中国电子产业快速发展之际。公司自创立之初便明确了其在电源管理领域的战略定位,专注于为市场提供高效、可靠的电源管理解决方案。通过深入研究市场需求和技术趋势,GStek迅速开发出包括线性稳压IC、PWM控制IC、电池保护IC及照明装置驱动IC在内的核心产品,奠定了其在行业内的技术基础。这一时期,GStek积极与国内外知名企业建立合作关系,不断拓展其市场份额。

问答坊 | AI 解惑

对面的高手看过来,ST公司的数据结构啥意思

typedef union {   u16 whole;   struct   {     unsigned IMPLEMENTED : 1; /**< Indicates that the key is implemented (from Customer Code to TSL only) */     unsigned ENABLED&nb ...…

查看全部问答>

复旦大学讲义-半导体器件原理PPT

一学期的课程 主要分为三个部分,半导体器件的工作原理 器件特性(MOSFET,BJT,VMOS,IGBT...GaAs) 半导体器件模型化 [ 本帖最后由 linda_xia 于 2010-4-18 08:31 编辑 ]…

查看全部问答>

一种利用MBE 模型改进的低速率

一种利用MBE 模型改进的低速率…

查看全部问答>

MC39I拨号GPRS成功分配IP后,无法进行UDP,TCP通信

我用的单片机控制MC39I,拨号CMNET成功分配IP地址和DNS1/DNS2后,发送UDP以及TCP数据包,为什么别人总收不到?当然上位机发的我也收不到.恳请那位大侠予以指点是不是在IPCP协商玩之后,发送UDP/TCP数据包之前还需要作些什么?我用Windows自带的PPP拨号发 ...…

查看全部问答>

关于PCB软件中的错误检测问题

在看一些书上的教程,或者是一些网上的文档的时候。都会介绍关于错误检查的问题。大家平常设计的时候修改这里修改的多吗?…

查看全部问答>

MSP430F2011发射38khz占空比为50%的方波红外线发射信号程序

用MSP430F2011发射38khz占空比为50%的方波红外线发射信号程序,谢谢邮箱com021@163.com,高手大侠们多多指教…

查看全部问答>

如何让管脚为输入状态且默认为高电平

我的按键扫描程序,无法检测到,原因是管脚电平一直为低。…

查看全部问答>

DIY建议+ADUCM361入门学习板

采用ADI最新的ARM CORTEX M3核的MCU 然后配上ADI的各种外设器件实现个入门的学习板…

查看全部问答>

我用CVAVR编译器,为什么编译后得不到COF和HEX文件?

我用CVAVR编译器,为什么编译后得不到COF和HEX文件?折腾了好几天了。请大伙帮帮忙看看是什么原因?先谢谢了! [ 本帖最后由 rjx 于 2012-10-10 08:23 编辑 ]…

查看全部问答>

STM32F429i的用途

1、拿到这块板子你会用它去做什么? 做个彩屏GUI手持通讯的工具 2、你会把你做好的东西发心得体会到论坛吗? 会,在论坛里与大家交流 3、你对这块板子的性价比怎么看?心里价位是多少? 性价比看价格了!这个比较矛盾的话题,量大了性价比就高! ...…

查看全部问答>