STM32 —— 多路ADC采集

#ifndef _ADC_H_
#define _ADC_H_

#include "type.h"
#include "debug.h"

void ADCInit(void);
uint16_t ADCGetData(uint8_t channel);
uint16_t ADCGetAverage(uint8_t channel, uint8_t times);

#ifndef CONFIG_CAN
void adc_test(void);
#endif

#endif /* _ADC_H_ */
// input1~5 -- 模拟量IO口; input6~10 -- 普通IO口用
#define ADC_CHANNEL        ADC1

#define AD_INPUT1_GRP      GPIOC
#define AD_INPUT1_INDEX    GPIO_Pin_0
#define AD_INPUT2_GRP      GPIOC
#define AD_INPUT2_INDEX    GPIO_Pin_1
#define AD_INPUT3_GRP      GPIOC
#define AD_INPUT3_INDEX    GPIO_Pin_2
#define AD_INPUT4_GRP      GPIOC
#define AD_INPUT4_INDEX    GPIO_Pin_3
#define AD_INPUT5_GRP      GPIOA
#define AD_INPUT5_INDEX    GPIO_Pin_2

#if 0
#define AD_INPUT6_GRP      GPIOA
#define AD_INPUT6_INDEX    GPIO_Pin_3
#define AD_INPUT7_GRP      GPIOC
#define AD_INPUT7_INDEX    GPIO_Pin_4
#define AD_INPUT8_GRP      GPIOC
#define AD_INPUT8_INDEX    GPIO_Pin_5
#define AD_INPUT9_GRP      GPIOB
#define AD_INPUT9_INDEX    GPIO_Pin_0
#define AD_INPUT10_GRP     GPIOB
#define AD_INPUT10_INDEX   GPIO_Pin_1
#endif
#define AD_INPUT_CONFIG(gpio, pos)  GPIOConfig(gpio, pos, GPIO_Mode_AIN)

#include "adc.h"
#include "stm32f10x.h"
#include "delay.h"
#include "target.h"

#define ADC_CHANNEL_NUM  5

static uint16_t ad_value[ADC_CHANNEL_NUM] = {0};

static void adc_gpio_clk_init(void)
{
  RCC_ADCCLKConfig(RCC_PCLK2_Div6);
}

static void adc_gpio_init(void)
{
  adc_gpio_clk_init();

  AD_INPUT_CONFIG(AD_INPUT1_GRP, AD_INPUT1_INDEX | AD_INPUT2_INDEX | AD_INPUT3_INDEX | AD_INPUT4_INDEX);
  AD_INPUT_CONFIG(AD_INPUT5_GRP, AD_INPUT5_INDEX);
  ADC_DeInit(ADC_CHANNEL);
}

static void adc_dma_init(void)
{
  DMA_InitTypeDef DMA_InitStructure;
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

  DMA_DeInit(DMA1_Channel1);

  DMA_InitStructure.DMA_PeripheralBaseAddr  = (u32) & (ADC1->DR);
  DMA_InitStructure.DMA_MemoryBaseAddr      = (u32)&ad_value;
  DMA_InitStructure.DMA_DIR                 = DMA_DIR_PeripheralSRC;
  DMA_InitStructure.DMA_M2M                 = DMA_M2M_Disable;
  DMA_InitStructure.DMA_PeripheralDataSize  = DMA_PeripheralDataSize_HalfWord;
  DMA_InitStructure.DMA_MemoryDataSize      = DMA_MemoryDataSize_HalfWord;
  DMA_InitStructure.DMA_BufferSize          = ADC_CHANNEL_NUM;
  DMA_InitStructure.DMA_MemoryInc           = DMA_MemoryInc_Enable;
  DMA_InitStructure.DMA_PeripheralInc       = DMA_PeripheralInc_Disable;
  DMA_InitStructure.DMA_Mode                = DMA_Mode_Circular;
  DMA_InitStructure.DMA_Priority            = DMA_Priority_High;
  DMA_Init(DMA1_Channel1, &DMA_InitStructure);
}

static void adc_init()
{
  ADC_InitTypeDef ADC_InitStructure;

  ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
  ADC_InitStructure.ADC_ScanConvMode = ENABLE;
  ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
  ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
  ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
  ADC_InitStructure.ADC_NbrOfChannel = ADC_CHANNEL_NUM;
  ADC_Init(ADC_CHANNEL, &ADC_InitStructure);

  ADC_RegularChannelConfig(ADC_CHANNEL, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);
  ADC_RegularChannelConfig(ADC_CHANNEL, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5);
  ADC_RegularChannelConfig(ADC_CHANNEL, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5);
  ADC_RegularChannelConfig(ADC_CHANNEL, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5);
  ADC_RegularChannelConfig(ADC_CHANNEL, ADC_Channel_4, 5, ADC_SampleTime_55Cycles5);

  ADC_DMACmd(ADC_CHANNEL, ENABLE);

  ADC_Cmd(ADC_CHANNEL, ENABLE);

  ADC_ResetCalibration(ADC_CHANNEL);

  while(ADC_GetResetCalibrationStatus(ADC_CHANNEL));

  ADC_StartCalibration(ADC_CHANNEL);

  while(ADC_GetCalibrationStatus(ADC_CHANNEL));
}

static void adc_start(void)
{
  ADC_SoftwareStartConvCmd(ADC_CHANNEL, ENABLE); // start convert
  DMA_Cmd(DMA1_Channel1, ENABLE);
}

void  ADCInit(void)
{
  adc_gpio_init();
  adc_dma_init();
  adc_init();
  adc_start();
}

uint16_t ADCGetData(uint8_t channel)
{
  uint16_t ret = 0;
  switch(channel)
  {
  case ADC_Channel_0:
    ret = ad_value[0];
    break;
  case ADC_Channel_1:
    ret = ad_value[1];
    break;
  case ADC_Channel_2:
    ret = ad_value[2];
    break;
  case ADC_Channel_3:
    ret = ad_value[3];
    break;
  case ADC_Channel_4:
    ret = ad_value[4];
    break;
  }

  return ret;
}

uint16_t ADCGetAverage(uint8_t channel, uint8_t times)
{
  uint16_t value;
  int i;

  for(i = 0; i < times; ++i)
  {
    value += ADCGetData(channel);
  }

  return (value / times);
}

#ifndef CONFIG_ADC
void adc_test(void)
{
  uint16_t buffer[ADC_CHANNEL_NUM];
  float temp[ADC_CHANNEL_NUM];

  buffer[0] = ADCGetData(ADC_Channel_0);
  temp[0] = (float)buffer[0] * 3.3 / 4096;
  printf("temp[0]: %f\r\n", temp[0]);

  buffer[1] = ADCGetData(ADC_Channel_1);
  temp[1] = (float)buffer[1] * 3.3 / 4096;
  printf("temp[1]: %f\r\n", temp[1]);

  buffer[2] = ADCGetData(ADC_Channel_2);
  temp[2] = (float)buffer[2] * 3.3 / 4096;
  printf("temp[2]: %f\r\n", temp[2]);

  buffer[3] = ADCGetData(ADC_Channel_1);
  temp[3] = (float)buffer[3] * 3.3 / 4096;
  printf("temp[1]: %f\r\n", temp[3]);

  buffer[4] = ADCGetData(ADC_Channel_4);
  temp[4] = (float)buffer[4] * 3.3 / 4096;
  printf("temp[2]: %f\r\n", temp[4]);
}
#endif

  • 6
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
STM32是一种基于ARM Cortex-M核心的32位微控制器系列,具有多个模拟到数字转换器(ADC)通道,可以同时采集多路信号。 STM32微控制器系列中的大多数型号都配备了多个ADC通道,可以同时采集不同的信号源。这些通道可以配置为单独使用或者以定时或事件触发的方式进行同步采集。 使用STM32多路ADC采集功能时,我们需要了解以下几个方面: 1. ADC通道配置:首先,我们需要确定采集的信号源数目,并选择合适的ADC通道进行配置。每个ADC通道都有自己的寄存器和控制位用于配置采样时间、分辨率和触发模式等参数。 2. 采样时间和时钟频率:采样时间指的是ADC模块在每次采样过程中对输入信号进行采样的持续时间。时钟频率则是ADC模块的工作频率,可以根据系统需求进行设置。 3. 触发模式:ADC可以通过定时器或外部事件触发进行采样。在多路ADC采集中,我们可以配置多个ADC通道使用同一个触发源进行同步采样。 4. 中断或DMA传输:采集到的数据可以通过中断或DMA传输到内存中,以便后续处理或分析。中断方式可以在每次采样结束后触发中断,而DMA传输可以在一次或多次采样结束后自动传输数据。 通过合理配置和使用STM32多路ADC采集功能,我们可以同时采集多个信号源,并根据实际需求进行相应的处理和分析。这在很多需要对多个信号同时进行监测和控制的应用中非常有用,例如工业自动化、仪器仪表和数据采集等领域。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值