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

发布时间:2022-12-16  

  在进行STM32F中AD采样的学习中,我们知道AD采样的方法有多种,按照逻辑程序处理有三种方式,一种是查询模式,一种是中断处理模式,一种是DMA模式。三种方法按照处理复杂方法DMA模式处理模式效率最高,其次是中断处理模式,最差是查询模式,相信很多学者在学习AD采样程序时,很多例程采用DMA模式,在这里我针对三种程序进行分别分析。

  1、AD采样查询模式

  在AD采样查询模式中,我们需要注意的是IO口的初始化配置,这里我采用PA2作为模拟采集的引脚(AIN2)和串口3作为打印输出。

  具体如下:建立一个USART3.C和USART3.H文件,其程序为:

  #include "usart3.h"

  #include "stdarg.h"

  u8 SendBuff[SENDBUFF_SIZE];

  void USART3_Config(void)

  {

  //定义结构体

  GPIO_InitTypeDef GPIO_InitStructure;

  USART_InitTypeDef USART_InitStructure;

  //开启外部时钟

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO,

  ENABLE);

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART3, ENABLE );

  // USART3 GPIO config

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOB, &GPIO_InitStructure);

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOAtiNG;

  GPIO_Init(GPIOB, &GPIO_InitStructure);

  //USART3 mode config

  USART_InitStructure.USART_BaudRate = 115200;

  USART_InitStructure.USART_WordLength = USART_WordLength_8b;

  USART_InitStructure.USART_StopBits = USART_StopBits_1;

  USART_InitStructure.USART_Parity = USART_Parity_No;

  USART_InitStructure.USART_HardwareFlowControl =

  USART_HardwareFlowControl_None;

  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

  USART_Init(USART3, &USART_InitStructure);

  USART_Cmd(USART3, ENABLE);

  }

  其次建立一个ADC.C和一个ADC.H文件,其中ADC.C中程序为:

  void ADC1_Init(void)

  {

  ADC1_GPIO_Config();

  ADC1_Mode_Config();

  }

  static void ADC1_GPIO_Config(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  //开启外部时钟

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 |

  RCC_APB2Periph_GPIOA,ENABLE);

  //配置PA2引脚

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;

  //配置为模拟输入

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

  //调用库函数

  GPIO_Init(GPIOA, &GPIO_InitStructure);

  }

  static void ADC1_Mode_Config(void)

  {

  //ADC1_ configuration

  ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;

  //独立ADC模式

  ADC_InitStructure.ADC_ScanConvMode = DISABLE;

  //禁止扫描模式

  ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;

  //开启连续转换模式

  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);

  //配置ADC时钟,为PCLK2的8分频,即9Mhz

  RCC_ADCCLKConfig(RCC_PCLK2_Div8);

  //配置ADC1的通道2位55.5个采集周期

  ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 1,

  ADC_SampleTime_55Cycles5);

  ADC_Cmd(ADC1,ENABLE);

  //复位校准寄存器

  ADC_ResetCalibration(ADC1);

  //等待校准寄存器复位完成

  while(ADC_GetResetCalibrationStatus(ADC1));

  //ADC校准

  ADC_StartCalibration(ADC1);

  while(ADC_GetCalibrationStatus(ADC1));

  //由于没有使用外部触发,所以使用软件触发ADC转换

  ADC_SoftwareStartConvCmd(ADC1,ENABLE);

  }

  然后在主函数main中其程序代码如下:

  int main(void)

  {

  USART3_Config();

  ADC1_Init();

  printf("输入ADC值");

  while(1)

  {

  ADC_ConvertedValue = ADC_GetConversionValue(ADC1);

  ADC_ConvertedValueLocal =(float)ADC_ConvertedValue/4096*3.3;

  //读取ADC转换的值

  printf("rn the current AD value = 0x%04X rn",ADC_ConvertedValue);

  printf("rn the current AD value = %f V

  rn",ADC_ConvertedValueLocal);

  Delay(0xFFFFEE);

  }

  }

  这样采用查询的方法即可以采集ADC的电压值,一个值为16进制转换的值,一个是转换计算的值。说明一下:ADC_ConvertedValue =

  ADC_GetConversionValue(ADC1);

  一定要放在while中,只有这样,采集的ADC电压值才是实时采集的电压值。放在while外面,则采集的电压值为第一次的电压值,且读取的电压值不会变化。对于4096的值来源在于ADC采集的数值是12位ADC,即是2的12次方。

  2、中断查询ADC程序

  对于中断查询采集ADC程序主要是在ADC.C和main函数中有差别。具体ADC.C程序为:

  void ADC1_Init(void)

  {

  ADC1_GPIO_Config();

  ADC1_Mode_Config();

  ADC_NVIC_Config();

  }

  static void ADC_NVIC_Config(void)

  {

  NVIC_InitTypeDef NVIC_InitStructure;

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

  NVIC_InitStructure.NVIC_IRQChannel = ADC1_2_IRQn;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  }

  static void ADC1_GPIO_Config(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  //开启外部时钟

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 |

  RCC_APB2Periph_GPIOA,ENABLE);

  //配置PA2引脚

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;

  //配置为模拟输入

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

  //调用库函数

  GPIO_Init(GPIOA, &GPIO_InitStructure);

  }

  static void ADC1_Mode_Config(void)

  {

  //ADC1_ configuration

  ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;

  //独立ADC模式

  ADC_InitStructure.ADC_ScanConvMode = DISABLE;

  //禁止扫描模式

  ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;

  //开启连续转换模式

  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);

  //配置ADC时钟,为PCLK2的8分频,即9Mhz

  RCC_ADCCLKConfig(RCC_PCLK2_Div8);

  //配置ADC1的通道2位55.5个采集周期

  ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 1,

  ADC_SampleTime_55Cycles5);

  ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE); //开启ADC采集中断

  ADC_Cmd(ADC1,ENABLE);

  //复位校准寄存器

  ADC_ResetCalibration(ADC1);

  //等待校准寄存器复位完成

  while(ADC_GetResetCalibrationStatus(ADC1));

  //ADC校准

  ADC_StartCalibration(ADC1);

  while(ADC_GetCalibrationStatus(ADC1));

  //由于没有使用外部触发,所以使用软件触发ADC转换

  ADC_SoftwareStartConvCmd(ADC1,ENABLE);

  }

  对于main函数如下:

  int main(void)

  {

  USART3_Config();

  ADC1_Init();

  printf("输入ADC值");

  while(1)

  {

  ADC_ConvertedValueLocal =(float)ADC_ConvertedValue/4096*3.3;

  //读取ADC转换的值

  printf("rn the current AD value = 0x%04X rn",ADC_ConvertedValue);

  printf("rn the current AD value = %f V

  rn",ADC_ConvertedValueLocal);

  Delay(0xFFFFEE);

  }

  }

  void ADC_IRQHandler(void)

  {

  IF (ADC_GetITStatus(ADC1, ADC_IT_EOC) == SET)

  {

  ADC_ConvertedValue = ADC_GetConversionValue(ADC1);

  }

  ADC_ClearITPendingBit(ADC1, ADC_IT_EOC);

  }

  在引入void ADC_IRQHandler(void)这个中断服务函数之前,一定要进行

  #define ADC_IRQHandler ADC1_2_IRQHandler

  否则中断无法执行,无法进行ADC采集。

  3、DMA模式的ADC采集程序

  采用这种方式的ADC采集程序,其在ADC.C程序为:

  void ADC1_Init(void)

  {

  ADC1_GPIO_Config();

  ADC1_Mode_Config();

  }

  static void ADC1_GPIO_Config(void)

  {

  GPIO_InitTypeDef GPIO_InitStructure;

  //开启外部时钟

  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 |

  RCC_APB2Periph_GPIOA,ENABLE);

  //配置PA2引脚

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;

  //配置为模拟输入

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;

  //调用库函数

  GPIO_Init(GPIOA, &GPIO_InitStructure);

  }

  static void ADC1_Mode_Config(void)

  {

  DMA_InitTypeDef DMA_InitStructure;

  ADC_InitTypeDef ADC_InitStructure;

  DMA_DeInit(DMA1_Channel1);

  DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;

  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;

文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

相关文章

    来评估模型的准确性、召回率、精确度等指标,并与未经过重采样的结果进行对比。 四、总结 非均匀数据重采样是解决非均匀数据集问题的重要步骤。通过过采样、欠采样、混合采样和加权重采样等方法,我们可以调整数据集的分布,提高......
    进行下一轮循环。 重采样的代码: p3 = [] index= int(random.random()*N) beta=0.0 mw=max(w) for i in range(N):     beta......
    关于STM32中AD采样的三种方法分析;  在进行STM32F中AD采样的学习中,我们知道AD采样的方法有多种,按照逻辑程序处理有三种方式,一种是查询模式,一种是中断处理模式,一种是DMA模式。三种方法按照处理复杂方法......
    STM32中AD采样的三种方法分析;  在进行STM32F中AD采样的学习中,我们知道AD采样的方法有多种,按照逻辑程序处理有三种方式,一种是查询模式,一种是中断处理模式,一种是DMA模式。三种方法按照处理复杂方法......
    交流电机调速的三种方法 只有两根线的电机是什么电机;  电动机调速的三种方法   电动机调速的三种方法如下:   机械调速法:通过机械传动装置改变电机的转速。如使用齿轮、皮带......
    在不同距离上的测量误差的对比情况,本文的方法在不同的距离上相对于其他三种方法测量误差都是最小的。表2列出了三种改进方法的在不同距离上的精度提高程度,在9个测量点上,比值法的精度提高程度平均为2.5%,而能......
    厂家的方案会略微修改一下,比如上图中T1/T2/T3,当T1 以上方法是比较稳妥,但是没有最大限度地输出有效电压,中间T1的零矢量是没有有效电压输出的。 本人提出三种方法,一种源自某半导体厂商,一种受到单电阻采样的......
    般说的频响曲线只是指幅频响应。 一个音频文件从手机里播放到被人听到需要经过哪些影响音质的过程。大致过程是这样的:音频文件-》操作系统的混音器(Mixer)-》操作系统DSP算法(音效、重采样,可能......
    在对鱼眼相机做立体几何重建之前需要进行有效的图像校准和畸变矫正。   当然,对于立体几何重建而言,通常会考虑保持极线笔直和水平的立体要求(主要满足立体几何中的对极几何要求),在此前提下,可能会利用类似最小化重采样的......
    于鱼眼[86]。鱼眼图像重采样的一个相关问题是,噪声函数被重采样过程扭曲,这对于任何试图最小化重投影误差的方法来说都是一个问题。Kukelova等人[73]使用标准视场相机的迭代技术解决了这一问题,该技......

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>