分享

Cubemx与HAL库系列教程|ADC DMA多通道采集详解

 myallmy 2021-10-29

什么是ADC

资料源码获取见文末

你以为的ADC

Image

哈哈,开个玩笑,S11 EDG LPL最后的荣光,加油~~~

说起来ADC,先来聊聊模拟信号与数字信号

模拟信号与数字信号简介

  • 模拟信号

模拟电压信号在时间上和幅值上均是连续的信号叫做模拟信号。此类信号的特点是,在一定动态范围内幅值可取任意值。

  • 数字信号

与模拟信号相对应,时间和幅值均离散( 不连续 ) 的信号叫做数字信号。数字信号的特点是幅值只可以取有限个值。

下文引自:

https://baijiahao.baidu.com/s?id=1701463655357562703&wfr=spider&for=pc

Image

通过观察声音的波形我们就会发现:

模拟信号在一段连续的时间范围内可以在任意时间点呈现任意数值,但是这种数值是随着时间连续变化。

Image

数字信号的取值是离散的二进制数,它和具有连续波形的模拟信号所展现出来的东西千差万别。

Image

模拟信号与数字信号如何相互转换

模拟信号一般通过脉码调制PCM的方法量化调制成数字信号,这个动作叫做模数转换。模拟信号会经过采样、量化、编码等一系列的动作最终转化成能被存储介质存储的一串0和1组成的数字信号。

也即是我们本次要介绍的ADC(analogue-to-digital conversion),模数转换。

Image

数字信号也可以还原成模拟信号。这种转化器件简称DAC,基本由即权电阻网络、运算放大器、基准电源和模拟开关组成。

Image

STM32 ADC介绍

上面说了一大堆,还是属于比较基础的介绍,有兴趣的小伙伴可以多多了解下AD转换器构成,实现原理,通信原理相信都学过,PCM编码都忘了吧,哈哈,好巧,我也忘了...

STM32 拥有 1~3 个 ADC(STM32F101/102 系列只有 1 个 ADC),这些 ADC 可以独立使用, 也可以使用双重模式(提高采样率)。

Image

STM32 的 ADC 是 12 位逐次逼近型的模拟数字转换器。它有 18 个通道,可测量 16 个外部和 2 个内部信号源。各通道的 A/D 转换可以单次、连续、扫 描或间断模式执行。ADC 的结果可以左对齐或右对齐方式存储在 16 位数据寄存器中。

STM32 的 ADC 最大的转换速率为 1Mhz,也就是转换时间为 1us(在 ADCCLK=14M,采样周期为 1.5 个 ADC 时钟下得到),不要让 ADC 的时钟超过 14M,否则将导致结果准确度下降

  • 转换时间

采样周期最小是 1.5 个,即如果我们要达到最快的采样,那么应该设置采样周期为 1.5 个周期,这里说的周期就是1/ADC_CLK

ADC 的总转换时间跟 ADC 的输入时钟和采样时间有关,其公式如下:

Tconv = 采样时间 + 12.5 个周期

其中 Tconv 为 ADC 总转换时间,当 ADC_CLK=14Mhz 的时候,并设置 1.5 个周期的采样时间,则 Tcovn=1.5+12.5=14 个周期=1us。通常经过 ADC 预分频器能分频到最大的时钟只能是 12M,采样周期设置为 1.5 个周期,算出最短的转换时间为 1.17us

外部的 16 个通道在转换的时候可分为 2 组通道:规则通道组和注入通道组,其中规则通道组最多有 16 路,注入通道组最多有 4 路

  • 规则通道组:

从名字来理解,规则通道就是一种规规矩矩的通道,类似于正常执行的程序。通常我们使用的都是这个通道

  • 注入通道组:

从名字来理解,注入即为插入,是一种不安分的通道,类似于中断。当程序正常往下执行时,中断可以打断程序的执行。同样如果在规则通道转换过程中,有注入通道插队,那么就要先转换完注入通道,等注入通道转换完成后,再回到规则通道的转换流程

  • DMA 请求

规则和注入通道转换结束后,除了产生中断外,还可以产生 DMA 请求,把转换好的数据直接存储在内存里面。

要注意的是只有 ADC1 和 ADC3 可以产生DMA 请求。一般我们在使用 ADC 的时候都会开启 DMA 传输

  • 转换方式

单次转换:顾名思义,ADC 执行一次转换。想要在转换需要再次开启

连续转换:

ADC 结束一个转换后立即启动一个新的转换,需要注意的是:此模式无法连续转换注入通道。连续模式下唯一的例外情况是,注入通道配置为在规则通道之后自动转换

STM ADC引脚映射

有些没有的管脚就不用关心了,比如F1的没有PF6-10引脚

ImageImage

DMA通道映射

Image

cubemx 配置

时钟之类的配置,劳烦各位小伙伴补补课,翻翻小飞哥前面的文章哈,直接进入正题

单通道DMA转换

时钟配置为分频之后为12MHZ

Image

选择 ADC1->IN8->PB0

Image

需要关注的几个点,扫描模式,这个在单通道时是无法使能的,只有多通道才可以开启,连续转换模式,根据自己实际需求决定是连续转换还是单次转换,触发方式,触发方式是非常多的,可以软件触发,PWM触发,定时器触发,也是根据自己的需要选择即可

Image

中断,需要就开启,不需要就不用开启,直白~Image

DMA配置,DMA的中断是默认开启的,并且无法配置关闭

ImageImage

配置很简单,你学废了吗...

代码实现

ADC配置的代码

Image

关于DMA的配置

Image
extern ADC_HandleTypeDef hadc1;
extern DMA_HandleTypeDef hdma_adc1;

uint16_t adc_buffer[50] = {0};
static void prvPrintTask( void *pvParameters )
{
 float adc_value = 0;
 
 HAL_ADCEx_Calibration_Start(&hadc1);
 HAL_ADC_Start_DMA(&hadc1,(uint32_t *)adc_buffer,50);
 int iIndexToString;
 /* Two instances of this task are created. The task parameter is used to pass
 an index into an array of strings into the task. Cast this to the required type. */
 iIndexToString = ( int ) pvParameters;
 for( ;; )
 {

 for(int index =0;index < 50;index++)
 {
  adc_value += adc_buffer[index];
 }
 adc_value/=50;
 
 adc_value = adc_value *3.3/(1<<12);
 debug_printf('\nadc_value = %.2f\n',adc_value);
  
 vTaskDelay( ( rand() & 0x1FF ) );
 }
}

配置为连续模式,DMA为循环模式,数据在buffer中不断循环更新Image

配置为不连续模式,只转换一次

Image

我是直接接到3.3V测试的,精度还可以Image

接GND,是有一些非零值的,所以必要的滤波还是要做的,这里我是用了最简单的均值滤波处理Image

多通道DMA转换

配置和单通道有些不同,扫描模式就可以打开了,通道数可以选择,我们选择4即可,下面的顺序就是我们要转换的顺序Image

/* ADC1 init function */
void MX_ADC1_Init(void)
{

  /* USER CODE BEGIN ADC1_Init 0 */

  /* USER CODE END ADC1_Init 0 */

  ADC_ChannelConfTypeDef sConfig = {0};

  /* USER CODE BEGIN ADC1_Init 1 */

  /* USER CODE END ADC1_Init 1 */
  /** Common config
  */
  hadc1.Instance = ADC1;
  hadc1.Init.ScanConvMode = ADC_SCAN_ENABLE;
  hadc1.Init.ContinuousConvMode = ENABLE;
  hadc1.Init.DiscontinuousConvMode = DISABLE;
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  hadc1.Init.NbrOfConversion = 4;
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }
  /** Configure Regular Channel
  */
  sConfig.Channel = ADC_CHANNEL_8;
  sConfig.Rank = ADC_REGULAR_RANK_1;
  sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /** Configure Regular Channel
  */
  sConfig.Channel = ADC_CHANNEL_9;
  sConfig.Rank = ADC_REGULAR_RANK_2;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /** Configure Regular Channel
  */
  sConfig.Channel = ADC_CHANNEL_10;
  sConfig.Rank = ADC_REGULAR_RANK_3;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /** Configure Regular Channel
  */
  sConfig.Channel = ADC_CHANNEL_11;
  sConfig.Rank = ADC_REGULAR_RANK_4;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN ADC1_Init 2 */

  /* USER CODE END ADC1_Init 2 */

}

主函数中编写如下代码:

为了更直观的观察效果,定义二维数组,每个存储4个值,可以看到,4个通道是依次转换的,各取10个值,求平均值

extern ADC_HandleTypeDef hadc1;
extern DMA_HandleTypeDef hdma_adc1;

uint16_t adc_buffer[10][4] = {0,0};
  
static void prvPrintTask( void *pvParameters )
{
 float adc_value[4] = {0};
 HAL_ADCEx_Calibration_Start(&hadc1);
 HAL_ADC_Start_DMA(&hadc1,(uint32_t *)adc_buffer,40);
 int iIndexToString;
 /* Two instances of this task are created. The task parameter is used to pass
 an index into an array of strings into the task. Cast this to the required type. */
 iIndexToString = ( int ) pvParameters;
 for( ;; )
 {
  for(int i=0;i<4;i++)
  {
   for(int j=0;j<10;j++)
   {
    adc_value[i]+=adc_buffer[j][i];
   }
  adc_value[i]=(float)adc_value[i]/(10*4096)*3.3;//求平均值并转换成电压值
  //ADC_Value[i]=(float)sum/10;
  printf('ADC_Value[%d] = %.2f\n',i,adc_value[i]);
 }  
  vTaskDelay( ( rand() & 0x1FF ) );
  }
}
Image

经验交流

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多