【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 */
}
相关推荐
懒惰的bit1 小时前
STM32F103C8T6板子使用说明
stm32·单片机·嵌入式硬件
芯岭技术2 小时前
PY32系列单片机离线烧录器,可配置选项字节和上机台批量烧录
单片机·嵌入式硬件
阿川!4 小时前
嵌入式软件--stm32 DAY7 I2C通讯上
stm32·单片机·嵌入式硬件·mcu
O。o.尊都假都5 小时前
socket套接字的超时控制
单片机·嵌入式硬件·网络协议
欢乐熊嵌入式编程6 小时前
智能手表项目的《项目背景与目标说明书》样本文档
嵌入式硬件·目标跟踪·规格说明书·智能手表
海尔辛7 小时前
学习黑客BitLocker与TPM详解
stm32·单片机·学习
海染棠花8 小时前
vscode+platformIO开发STM32(八)
ide·vscode·stm32
上海合宙LuatOS8 小时前
全栈工程师实战手册:LuatOS日志系统开发指南!
java·开发语言·单片机·嵌入式硬件·物联网·php·硬件工程
公子无缘8 小时前
【嵌入式】记一次解决VScode+PlatformIO安装卡死的经历
vscode·stm32·单片机·mcu·platformio
昊昊昊昊昊明8 小时前
十天学会嵌入式技术之51单片机—day-9
单片机·嵌入式硬件·51单片机