【STM32】ADC|多通道ADC采集

本次实现的是ADC实现数字信号与模拟信号的转化,数字信号时不连续的,模拟信号是连续的。

1.ADC转化的原理

模拟-数字转换技术使用的是逐次逼近法,使用二分比较的方法来确定电压值

当单片机对应的参考电压为3.3v时,0~ 3.3v(模拟信号)对应0~4095(数字信号),假如我们知道一个数字信号的值为x,那么他对应的模拟信号

x/4095=y/3.3

y就是x对应的模拟信号

假如要确定0.8v对应的数字信号的值

他会将0.8v电压保存在电容中,然后先和3.3v的一半对应的1.65v(2048)做比较,如果小于1.65v的话,12位最高位就是0,因为1000 0000 0000 对应的就是2048

接着就是和1.65的一半比较,0.825v(1024)>0.8v,所以12位中倒数第二高位就是0,因为0.825(1024)对应二进制就是0100 0000 0000

接着0.8v和0.825的一半比较->0.4125v(512),0.8v>0.4125v,所以第三位为1,因为512对应的二进制为0010 00000000

按这个方法依次确定0.8v对应的12位的二进制数,然后这个二进制对应的十进制就是0.8v对应的数字信号的值

所以对应的过程是1.启动ADC 2.采样&转换 3.获取&计算,针对STM32F103C8T6芯片有10个外部通道,和两个内部通道(内部温度传感器,内部参考电压)进行ADC的转化,ADC1和ADC2,两个转化结构,每个转化结构都有一个注入组和一个规则组,我们现在只讲讲规则组,我们要对一个通道的电压值进行采集的时候,我们需要将这个通道注册进这个通道中,当启动ADC,然后采样转化的电压值就会放在规则通道数据寄存器(12位二进制值)中,等待获取

2.实操

单片机有一个电位器,通过旋转电位器,使得PA5的输出电压发送变化,我们将其输出到串口上去

对应的PA5刚好是ADC1的通道5

c 复制代码
  char message[50]="";
  float v=0.0;
  int value=0;
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  { 
	  HAL_ADC_Start(&hadc1);//开启ADC1
	  HAL_ADC_PollForConversion(&hadc1,HAL_MAX_DELAY);//等待转换
	  value=HAL_ADC_GetValue(&hadc1);//从寄存器中获取电压值
	  v=(value/4095.0)*3.3;//转模拟信号
	  sprintf(message,"v:%.2f,value:%d",v,value);
	  HAL_UART_Transmit(&huart2,message,strlen(message),HAL_MAX_DELAY);
	  HAL_Delay(500);
   /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }

但是发现一个问题,就是电位器扭到头

最大为3.25v,达不到3.3v,这是因为手册里面说

c 复制代码
  char message[50]="";
  float v=0.0;
  int value=0;
  HAL_ADCEx_Calibration_Start(&hadc1);//校准
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
	  HAL_ADC_Start(&hadc1);//开启ADC1
	  HAL_ADC_PollForConversion(&hadc1,HAL_MAX_DELAY);//等待转换
	  value=HAL_ADC_GetValue(&hadc1);//从寄存器中获取电压值
	  v=(value/4095.0)*3.3;//转模拟信号
	  sprintf(message,"v:%.2f,value:%d",v,value);
	  HAL_UART_Transmit(&huart2,(uint8_t*)message,strlen(message),HAL_MAX_DELAY);
	  HAL_Delay(500);




    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }

其实还有一种是循环转换,只需要开启一次ADC即可

c 复制代码
  char message[50]="";
  float v=0.0;
  int value=0;
  HAL_ADCEx_Calibration_Start(&hadc1);//校准
  HAL_ADC_Start(&hadc1);//开启ADC1
  HAL_ADC_PollForConversion(&hadc1,HAL_MAX_DELAY);//等待转换
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
	
	  value=HAL_ADC_GetValue(&hadc1);//从寄存器中获取电压值
	  v=(value/4095.0)*3.3;//转模拟信号
	  sprintf(message,"v:%.2f,value:%d",v,value);
	  HAL_UART_Transmit(&huart2,(uint8_t*)message,strlen(message),HAL_MAX_DELAY);
	  HAL_Delay(500);




    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }

3.ADC多路采集

在前面说过,如果要获取对应通道的电压值,就需要将这个通道注册进规则组中,然后数据准备就绪, HAL_ADC_PollForConversion函数不断的检测ADC状态寄存器中转换结束标志位(EOC)是否为1,如果是,转换完成,将转换的值放入规则通道数据寄存器中,然后调用HAL_ADC_GetValue读取规则通道数据寄存器,然后对应的ADC状态寄存器中的转换结束标志位置0.

在多采集中需要将多个通道打开实现转换,会按照规则组注册顺序,依次将对应通道的电压值放入规则通道数据寄存器,然后写到内存中。

我们可以使用DMA来对规则通道数据寄存器的值搬运到内存,当搬运完成,就会触发DMA完成中断,进行处理转换后的值

本次实现热敏电阻,电位器,单片机内部温度,内部参考电压的ADC转换,并通过串口将其发送


PA4,PA5 ADC在ADC1的通道4,通道5上,以及内部温度通道,内部参考电压通道

注意将四个通道的多久采集值调到四个通道转换最大值,确保数据都已经被转换

c 复制代码
uint16_t data[4];
char message[50]="";
void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc)
{

  if(hadc==&hadc1)
  {
	  sprintf(message,"%d %d %d %d",data[0],data[1],data[2],data[3]);
      HAL_UART_Transmit(&huart2, (uint8_t*)message,sizeof(message),HAL_MAX_DELAY);

  }

}
int main(void)
{

  HAL_Init();



 

  HAL_ADCEx_Calibration_Start(&hadc1);//校准

  while (1)
  {

	  HAL_ADC_Start_DMA(&hadc1,(uint32_t*)data,4);//开启ADC,DMA完成搬运调用DMA完成中断中断,在中断中发送转换值到串口
	  HAL_Delay(500);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

依次是电位器,热敏电阻,内部温度,内部参考电压未转化为模拟电压

除此之外我们可以使用循环转换,和DMA的循环搬运就可以实现开启一次ADC就可以一直将转换后的四个值发送到串口,因为会一直转换,不用知道什么时候转换完成,我们将发送到串口放到while中

c 复制代码
uint16_t data[4];
char message[50]="";

int main(void)
{

  HAL_Init();



 

  HAL_ADCEx_Calibration_Start(&hadc1);//校准
 HAL_ADC_Start_DMA(&hadc1,(uint32_t*)data,4);//开启ADC,DMA完成搬运调用DMA完成中断中断,在中断中发送转换值到串口
  while (1)
  {

	  
	  sprintf(message,"%d %d %d %d",data[0],data[1],data[2],data[3]);
      HAL_UART_Transmit(&huart2, (uint8_t*)message,sizeof(message),HAL_MAX_DELAY);
	  HAL_Delay(500);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}
相关推荐
SY师弟11 分钟前
51单片机基础部分——矩阵按键检测
嵌入式硬件·矩阵·51单片机
Lester_11011 小时前
嵌入式学习笔记 - freeRTOS xTaskResumeAll( )函数解析
笔记·stm32·单片机·学习·freertos
阿超爱嵌入式2 小时前
STM32学习之I2C(理论篇)
stm32·嵌入式硬件·学习
三月雪落无痕4 小时前
altium designer2024绘制stm32过程笔记x`
笔记·嵌入式硬件
天天爱吃肉82186 小时前
新能源汽车热管理核心技术解析:冬季续航提升40%的行业方案
android·python·嵌入式硬件·汽车
章鱼哥嵌入式开发6 小时前
# STM32F103 SD卡读写程序
stm32·单片机
广药门徒7 小时前
定时器时钟来源可以从输入捕获引脚输入
单片机·嵌入式硬件
爱睡觉的王宇昊11 小时前
二、【ESP32开发全栈指南:ESP32 GPIO深度使用】
单片机·嵌入式硬件
学生哥-_-17 小时前
STM32通过KEIL pack包轻松移植LVGL,并学会使用GUI guider
stm32·lvgl·tftlcd·gui guider·gt911
三三十二17 小时前
STM32实战:数字音频播放器开发指南
stm32·单片机·嵌入式硬件