STM32F4XX的12位ADC采集数值超过4096右对齐模式设置失败

文章目录

  • 一、前言
  • 二、问题1:数值超过4096
  • 三、问题1的排错过程
  • 四、问题2:右对齐模式设置失败
  • 五、问题2的解决方法
    • 5.1 将ADC_ExternalTrigConv设置为0
    • 5.2 使用ADC_StructInit()函数

一、前言

最近在学习STM32的ADC功能,遇到了一个奇怪的问题。
使用芯片:STM32F407ZGT6
使用函数:库函数
使用代码:正点原子的例程《实验16 ADC实验》
串口工具:VOFA

二、问题1:数值超过4096

博主直接使用了正点原子的程序,如下面所示,使用的12位的ADC1,端口是PA5

//初始化ADC															   
void  Adc_Init(void)
{    GPIO_InitTypeDef  GPIO_InitStructure;ADC_CommonInitTypeDef ADC_CommonInitStructure;ADC_InitTypeDef       ADC_InitStructure;RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能GPIOA时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //使能ADC1时钟//先初始化ADC1通道5 IO口GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;//PA5 通道5GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;//模拟输入GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;//不带上下拉GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化  RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,ENABLE);	  //ADC1复位RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,DISABLE);	//复位结束	 ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;//两个采样阶段之间的延迟5个时钟ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //DMA失能ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;//预分频4分频。ADCCLK=PCLK2/4=84/4=21Mhz,ADC时钟最好不要超过36Mhz ADC_CommonInit(&ADC_CommonInitStructure);//初始化ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//12位模式ADC_InitStructure.ADC_ScanConvMode = DISABLE;//非扫描模式	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//关闭连续转换ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;//禁止触发检测,使用软件触发ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐	ADC_InitStructure.ADC_NbrOfConversion = 1;//1个转换在规则序列中 也就是只转换规则序列1 ADC_Init(ADC1, &ADC_InitStructure);//ADC初始化ADC_Cmd(ADC1, ENABLE);//开启AD转换器	
}

为了容易查看数值,博主将adc.h和adc.c移植到了串口打印的程序之中,这样子可以在电脑的串口工具查看相对于的数值。
主函数的代码如下,将ADC采集到的原始数值和转化为电压之后的数值用串口打印出来

adcx=Get_Adc_Average(ADC_Channel_5,20);//获取通道5的转换值,20次取平均
temp=(float)adcx*(3.3/4096);          //获取计算后的带小数的实际电压值,比如3.1111
printf("%d,%f\n",adcx,temp);

从下面的结果可以看出超过我们的12位ADC的最大值4096(2^12=4096)。

在这里插入图片描述

三、问题1的排错过程

首先,博主将ADC检测引脚接到3.3V,出现了下面的状况:ADC采集到的数值很大,接近于65535,也就是2^16。

在这里插入图片描述
这让博主想到应该是ADC设置模型的时候设置为左对齐模型,查看了代码(如下)后发现设置的是右对齐模式

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐

根据《STM32F407参考手册》P255可以得到下面结论:

  • 右对齐:将采集到的数据放在低12位
  • 左对齐:将采集到的数据放在高12位

在这里插入图片描述

为了验证是否真的是左对齐,博主先假设其位左对齐,根据上面的知识可以知道,想要实现右对齐,只需要将采集到的数值向右移动四位即可,因此,博主将主函数改为如下的代码:

adcx=Get_Adc_Average(ADC_Channel_5,20);//获取通道5的转换值,20次取平均
adcx = adcx >> 4;
temp=(float)adcx*(3.3/4096);          //获取计算后的带小数的实际电压值,比如3.1111
printf("%d,%f\n",adcx,temp);

结果如下图所示,左边是浮空状态下的数值,右边是将ADC检测引脚接到3.3V的数值,可以看出结果小于4096。

在这里插入图片描述
综上可得:虽然代码设置了右对齐,但是实际上是左对齐。

四、问题2:右对齐模式设置失败

为了探究对齐模式设置失败的问题,博主先学习了ADC的寄存器,在《STM32F407参考手册》P276里面找到对齐模式的设置存在于寄存器ADC_CR2里面的第11位,名称是ALIGN
在这里插入图片描述
在这里插入图片描述

打开Keil的调试模式,查看到代码运行过ADC的初始化ADC_Init(ADC1, &ADC_InitStructure)之后,低11位的ALIGN居然是打勾的,也就是1,代表就是左对齐!!!

在这里插入图片描述

为了进一步研究问题所在,博主进入函数ADC_Init(),查看到给CR2(就是前面学习到设置对齐模式的寄存器)赋值的是tmpreg1 ,可以看到tmpreg1和四个值进行了 或操作 ,根据之前的博客《STM32需要的基本知识——位运算》可知,如果这四个值的第11位是1的话,那么或操作之后整个CR2的第11位就是1。

  tmpreg1 |= (uint32_t)(ADC_InitStruct->ADC_DataAlign | \ADC_InitStruct->ADC_ExternalTrigConv | ADC_InitStruct->ADC_ExternalTrigConvEdge | \((uint32_t)ADC_InitStruct->ADC_ContinuousConvMode << 1));/* Write to ADCx CR2 */ADCx->CR2 = tmpreg1;

接下来便进一步调试查看这四个值的数值,查看对应的二进制:

  • ADC_DataAlign:0x00000000
  • ADC_ExternalTrigConv:0x0501BD00
  • ADC_ExternalTrigConvEdge :0x00000000
  • ADC_ContinuousConvMode:0x00

在这里插入图片描述
ADC_ContinuousConvMode << 1 = 0x00 << 1 = 0x00
ADC_ExternalTrigConv:0x0501BD00= 0101 0001 0000 1011 1101 0000 0000
可以看到第11位是1,那么根据上面的结论可以知道tmpreg1的第11位也是1,即整个CR2的第11位就是1。
也就是说是因为ADC_ExternalTrigConv导致我们设置的对齐模式位左对齐。
进一步回顾正点原子的ADC初始化代码(如下)可以发现,确实没有对这一项进行初始化。
至于为什么他没有初始化,这个博主就不得而知了。

  ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;//两个采样阶段之间的延迟5个时钟ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //DMA失能ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;//预分频4分频。ADCCLK=PCLK2/4=84/4=21Mhz,ADC时钟最好不要超过36Mhz ADC_CommonInit(&ADC_CommonInitStructure);//初始化ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//12位模式ADC_InitStructure.ADC_ScanConvMode = DISABLE;//非扫描模式	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//关闭连续转换ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;//禁止触发检测,使用软件触发ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐	ADC_InitStructure.ADC_NbrOfConversion = 1;//1个转换在规则序列中 也就是只转换规则序列1 ADC_Init(ADC1, &ADC_InitStructure);//ADC初始化

综上可得:右对齐模式设置失败的原因是没对ADC_InitStructure.ADC_ExternalTrigConv进行初始化设置。

五、问题2的解决方法

知道了问题所在,博主目前研究两种解决方法:
第一种:将ADC_ExternalTrigConv设置为0;
第二种:使用ADC_StructInit(ADC_InitTypeDef* ADC_InitStruct);初始化ADC_ExternalTrigConv;

5.1 将ADC_ExternalTrigConv设置为0

既然是ADC_ExternalTrigConv不为零导致的,那直接将其设置为0,即添加代码

  ADC_InitStructure.ADC_ExternalTrigConv = 0;//添加这一行

ADC的完整代码如下:

//初始化ADC															   
void  Adc_Init(void)
{    GPIO_InitTypeDef  GPIO_InitStructure;ADC_CommonInitTypeDef ADC_CommonInitStructure;ADC_InitTypeDef       ADC_InitStructure;RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能GPIOA时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //使能ADC1时钟//先初始化ADC1通道5 IO口GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;//PA5 通道5GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;//模拟输入GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;//不带上下拉GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化  RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,ENABLE);	  //ADC1复位RCC_APB2PeriphResetCmd(RCC_APB2Periph_ADC1,DISABLE);	//复位结束	 ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;//两个采样阶段之间的延迟5个时钟ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //DMA失能ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;//预分频4分频。ADCCLK=PCLK2/4=84/4=21Mhz,ADC时钟最好不要超过36Mhz ADC_CommonInit(&ADC_CommonInitStructure);//初始化ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//12位模式ADC_InitStructure.ADC_ScanConvMode = DISABLE;//非扫描模式	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//关闭连续转换ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;//禁止触发检测,使用软件触发ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐	ADC_InitStructure.ADC_NbrOfConversion = 1;//1个转换在规则序列中 也就是只转换规则序列1 //====================================================//ADC_InitStructure.ADC_ExternalTrigConv = 0;//添加这一行//====================================================//ADC_Init(ADC1, &ADC_InitStructure);//ADC初始化ADC_Cmd(ADC1, ENABLE);//开启AD转换器	
}

5.2 使用ADC_StructInit()函数

从上面的代码可以看出 ADC_ExternalTrigConv 是结构 ADC_InitTypeDef 的一个成员,那看看有没有官方自带的初始化函数。

ADC_InitTypeDef       ADC_InitStructure;
ADC_InitStructure.ADC_ExternalTrigConv = 0

stm32f4xx_adc.h里面看到有对结构体进行初始化的函数,且有对 ADC_InitTypeDef 进行初始化的。

/* Initialization and Configuration functions *********************************/
void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);
void ADC_StructInit(ADC_InitTypeDef* ADC_InitStruct);
void ADC_CommonInit(ADC_CommonInitTypeDef* ADC_CommonInitStruct);
void ADC_CommonStructInit(ADC_CommonInitTypeDef* ADC_CommonInitStruct);
void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);

stm32f4xx_adc.c进一步研究该函数的代码:

/*** @brief  Fills each ADC_InitStruct member with its default value.* @note   This function is used to initialize the global features of the ADC ( *         Resolution and Data Alignment), however, the rest of the configuration*         parameters are specific to the regular channels group (scan mode *         activation, continuous mode activation, External trigger source and *         edge, number of conversion in the regular channels group sequencer).  * @param  ADC_InitStruct: pointer to an ADC_InitTypeDef structure which will *         be initialized.* @retval None*/
void ADC_StructInit(ADC_InitTypeDef* ADC_InitStruct)
{/* Initialize the ADC_Mode member */ADC_InitStruct->ADC_Resolution = ADC_Resolution_12b;/* initialize the ADC_ScanConvMode member */ADC_InitStruct->ADC_ScanConvMode = DISABLE;/* Initialize the ADC_ContinuousConvMode member */ADC_InitStruct->ADC_ContinuousConvMode = DISABLE;/* Initialize the ADC_ExternalTrigConvEdge member */ADC_InitStruct->ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;/* Initialize the ADC_ExternalTrigConv member */ADC_InitStruct->ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC1;/* Initialize the ADC_DataAlign member */ADC_InitStruct->ADC_DataAlign = ADC_DataAlign_Right;/* Initialize the ADC_NbrOfConversion member */ADC_InitStruct->ADC_NbrOfConversion = 1;
}

其中的 ADC_ExternalTrigConv_T1_CC1 确实设置的是0。

#define ADC_ExternalTrigConv_T1_CC1                ((uint32_t)0x00000000)
#define ADC_ExternalTrigConv_T1_CC2                ((uint32_t)0x01000000)
#define ADC_ExternalTrigConv_T1_CC3                ((uint32_t)0x02000000)
#define ADC_ExternalTrigConv_T2_CC2                ((uint32_t)0x03000000)
#define ADC_ExternalTrigConv_T2_CC3                ((uint32_t)0x04000000)
#define ADC_ExternalTrigConv_T2_CC4                ((uint32_t)0x05000000)
#define ADC_ExternalTrigConv_T2_TRGO               ((uint32_t)0x06000000)
#define ADC_ExternalTrigConv_T3_CC1                ((uint32_t)0x07000000)
#define ADC_ExternalTrigConv_T3_TRGO               ((uint32_t)0x08000000)
#define ADC_ExternalTrigConv_T4_CC4                ((uint32_t)0x09000000)
#define ADC_ExternalTrigConv_T5_CC1                ((uint32_t)0x0A000000)
#define ADC_ExternalTrigConv_T5_CC2                ((uint32_t)0x0B000000)
#define ADC_ExternalTrigConv_T5_CC3                ((uint32_t)0x0C000000)
#define ADC_ExternalTrigConv_T8_CC1                ((uint32_t)0x0D000000)
#define ADC_ExternalTrigConv_T8_TRGO               ((uint32_t)0x0E000000)
#define ADC_ExternalTrigConv_Ext_IT11              ((uint32_t)0x0F000000)

所以加一行对结构体的初始化即可:

ADC_StructInit(&ADC_InitStructure);

完整代码如下:

//初始化ADC															   
void  Adc_Init(void)
{    GPIO_InitTypeDef  GPIO_InitStructure;ADC_CommonInitTypeDef ADC_CommonInitStructure;ADC_InitTypeDef       ADC_InitStructure;RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能GPIOA时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //使能ADC1时钟//先初始化ADC1通道5 IO口GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;//PA5 通道5GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AN;//模拟输入GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ;//不带上下拉GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化  ADC_DeInit();//ADC复位ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;//独立模式ADC_CommonInitStructure.ADC_TwoSamplingDelay = ADC_TwoSamplingDelay_5Cycles;//两个采样阶段之间的延迟5个时钟ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled; //DMA失能ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div4;//预分频4分频。ADCCLK=PCLK2/4=84/4=21Mhz,ADC时钟最好不要超过36Mhz ADC_CommonInit(&ADC_CommonInitStructure);//初始化ADC_StructInit(&ADC_InitStructure);ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;//12位模式ADC_InitStructure.ADC_ScanConvMode = DISABLE;//非扫描模式	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;//关闭连续转换ADC_InitStructure.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None;//禁止触发检测,使用软件触发ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//右对齐	ADC_InitStructure.ADC_NbrOfConversion = 1;//1个转换在规则序列中 也就是只转换规则序列1 ADC_Init(ADC1, &ADC_InitStructure);//ADC初始化ADC_Cmd(ADC1, ENABLE);//开启AD转换器	
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/336102.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#PDF转Excel

組件 Spire.Pdf.dll, v7.8.9.0 【注意&#xff1a;版本太低的没有此功能】 在Visual Studio中找到参考&#xff0c;鼠标右键点击“引用”&#xff0c;“添加引用”&#xff0c;将本地路径debug文件夹下的dll文件添加引用至程序。 界面图&#xff1a; 1个label&#xff0c;1…

酷开科技凭借AIGC技术打造从产品到运营到生态的范本

近日&#xff0c;酷开科技成功挑战“全球最多人同时线上和线下开箱”吉尼斯纪录&#xff0c;为中国品牌出海打样。酷开科技&#xff0c;除了硬件上的实力&#xff0c;更有软件上的硬核。酷开科技之所以能够从中国OTT行业独角兽走向海外市场“开疆拓土”&#xff0c;是基于创新的…

echarts使用之柱状图

一、引入Echarts npm install eacharts --save 二、选择一个Echarts图 选择创建一个柱状图 option { // x轴参数的基本配置xAxis: {type: category,data: [Mon, Tue, Wed, Thu, Fri, Sat, Sun] //X轴数据}, // y轴参数的基本配置yAxis: {type: value}, // series:[{data: …

【HarmonyOS】掌握 Stage 模型的核心概念与应用

从今天开始&#xff0c;博主将开设一门新的专栏用来讲解市面上比较热门的技术 “鸿蒙开发”&#xff0c;对于刚接触这项技术的小伙伴在学习鸿蒙开发之前&#xff0c;有必要先了解一下鸿蒙&#xff0c;从你的角度来讲&#xff0c;你认为什么是鸿蒙呢&#xff1f;它出现的意义又是…

使用Android Compose实现网格列表滑到底部的提示信息展示

文章目录 概述1 效果对比1.1 使用添加Item的办法&#xff1a;1.2 使用自定义的方法 2. 效果实现2.1 列表为空时的提示页面实现2.2 添加Item的方式代码实现2.3 使用自定义的方式实现 3. UI工具类 概述 目前大多数的APP都会使用列表的方式来呈现内容&#xff0c;例如淘宝&#x…

Fotor免费图片编辑,怎么使用Pro版本

Fotor是一款多功能的图像处理工具&#xff0c;它提供了丰富的图片编辑和设计功能。以下是Fotor的一些主要特点和功能&#xff1a; 基本编辑工具&#xff1a; Fotor允许用户进行常见的基本编辑&#xff0c;如裁剪、旋转、调整亮度、对比度和饱和度等。 滤镜和效果&#xff1a; …

来聊聊守护线程

写在文章前面 近期和朋友交流时谈及守护线程&#xff0c;遂整理一篇关于守护线程的文章&#xff0c;让读者会守护线程有更进一步的认识&#xff0c;本文整体内容如下&#xff1a; 你好&#xff0c;我叫sharkchili&#xff0c;目前还是在一线奋斗的Java开发&#xff0c;经历过很…

GCF:在线市场异质治疗效果估计的广义因果森林

英文题目&#xff1a;GCF: Generalized Causal Forest for Heterogeneous Treatment Effects Estimation in Online Marketplace 中文题目&#xff1a;GCF&#xff1a;在线市场异质治疗效果估计的广义因果森林 单位&#xff1a;滴滴&美团 时间&#xff1a;2022 论文链接…

关于TLS相关安全配置问题

近期被信安部门反馈了TLS几个安全漏洞。 SSL/TLS协议信息泄露漏洞(CVE-2016-2183)【原理扫描】SSL/TLS 受诫礼(BAR-MITZVAH)攻击漏洞(CVE-2015-2808)【原理扫描】SSL/TLS 服务器瞬时 Diffie-Hellman 公共密钥过弱【原理扫描】SSL/TLS RC4 信息泄露漏洞(CVE-2013-2566)【原理扫…

PINN物理信息网络 | 泊松方程的物理信息神经网络PINN解法

基本介绍 泊松方程是一种常见的偏微分方程&#xff0c;它在物理学和工程学中具有广泛的应用。它描述了在某个区域内的标量场的分布与该场在该区域边界上的值之间的关系。 物理信息神经网络&#xff08;PINN&#xff09;是一种结合了物理定律和神经网络的方法&#xff0c;用于…

2D和3D视觉技术有哪些不同特点?

​作为一个多年经验的机器视觉工程师&#xff0c;我将详细介绍2D和3D视觉技术的不同特点、应用场景以及它们能够解决的问题。在这个领域内&#xff0c;2D和3D视觉技术是实现自动化和智能制造的关键技术&#xff0c;它们在工业检测、机器人导航、质量控制等众多领域都有着广泛的…

无心剑中译塞缪尔·厄尔曼《青春》

Youth 青春 Samuel Ullman 塞缪尔厄尔曼 Youth is not a time of life; it is a state of mind; it is not a matter of rosy cheeks, red lips and supple knees; it is a matter of the will, a quality of the imagination, a vigor of the emotions; it is the freshness o…