单片机内部ADC+DMA实现多路模拟信号采集(基于GD32F103)

市面上常见的32位CM3/CM4内核的单片机通常自带12位精度的ADC外设,可以实现采集精度要求不高的应用场景。

原理:

精度12位,意思是将输入信号转换为12位的二进制数,对应到十进制即2的12次方,就是4096。

一般的MCU的参考电压是3.3V,意味着在将待测电压直接加到单片机引脚的情况下,可以采集0-3.3V之间的电压信号。

当输入电压为0时,转换后的值为0,当输入电压为3.3V时,转换后的值为4095。

计算方式很简单,计算公式:待测电压x / 参考电压3.3 = 转换后的AD码值 / 4095。

下面给出代码。

bsp_adc.h文件中包含外部变量及函数声明。

#ifndef _BSP_ADC_H
#define _BSP_ADC_H
#include "gd32f10x.h"

extern uint16_t	adc_value[2];

void bsp_init_adc(void);
void adc_config(void);
void adc_dma_config(void);


#endif

bsp_adc.c中包含具体的函数实现,代码注释已经解释较为清晰,不再赘述。此处用PA1和PA3实现两路采集。

#include "bsp_adc.h"
#include "systick.h"

/* 初始化ADC采集 */
void bsp_init_adc(void)
{
	adc_config();		/* 配置GPIO及ADC */
	adc_dma_config();	/* 配置DMA */
}

/* 配置GPIO及ADC */
void adc_config(void)
{
    /* 使能IO口时钟 */
    rcu_periph_clock_enable(RCU_GPIOA);
	
    /* 使能ADC0时钟 */
    rcu_periph_clock_enable(RCU_ADC0);
    /* 配置ADC时钟 */
    rcu_adc_clock_config(RCU_CKADC_CKAPB2_DIV8);   /* 8分频后为108/8=13.5M */
	
    /* 配置对应IO口 */
    gpio_init(GPIOA, GPIO_MODE_AIN, GPIO_OSPEED_50MHZ, GPIO_PIN_1); 
    gpio_init(GPIOA, GPIO_MODE_AIN, GPIO_OSPEED_50MHZ, GPIO_PIN_3); 

   /* 复位ADC */
    adc_deinit(ADC0);
	
    /* 配置ADC独立模式*/
    adc_mode_config(ADC_MODE_FREE); 
	
    /* 配置ADC连续转换模式*/
    adc_special_function_config(ADC0, ADC_CONTINUOUS_MODE, ENABLE); 
    /* 配置ADC扫描模式,多路采集需要采用扫描模式*/
    adc_special_function_config(ADC0, ADC_SCAN_MODE, ENABLE);

    /* 配置ADC数据对齐方式为右对齐*/
    adc_data_alignment_config(ADC0, ADC_DATAALIGN_RIGHT);
	
    /* 配置ADC规则通道组长度,采集几路就配置为几*/
    adc_channel_length_config(ADC0, ADC_REGULAR_CHANNEL, 2);
	
	/* 配置ADC规则组,注意这个位置的rank必须从0开始*/
    adc_regular_channel_config(ADC0, 0, ADC_CHANNEL_1, ADC_SAMPLETIME_55POINT5);
    adc_regular_channel_config(ADC0, 1, ADC_CHANNEL_3, ADC_SAMPLETIME_55POINT5);

    /* 配置ADC软件触发转换*/
    adc_external_trigger_source_config(ADC0, ADC_REGULAR_CHANNEL, ADC0_1_2_EXTTRIG_REGULAR_NONE);

    /* 使能转换*/
    adc_external_trigger_config(ADC0, ADC_REGULAR_CHANNEL, ENABLE);
	
    /* 使能ADC */
    adc_enable(ADC0); 
    delay_1ms(1);
    /* 使能ADC校准*/
    adc_calibration_enable(ADC0);	
	
    /* 使能ADC的DMA传输 */
    adc_dma_mode_enable(ADC0);
	
	/* 使能软件转换 */
    adc_software_trigger_enable(ADC0, ADC_REGULAR_CHANNEL);
}

/* 配置ADC对应的DMA */
/* GD32F103的ADC0对应DMA0的通道0,用户根据实际情况调整 */
void adc_dma_config(void)
{
	/* 定义DMA参数结构体变量 */
	dma_parameter_struct dma_init_struct;
	
    /* 使能DMA时钟 */
    rcu_periph_clock_enable(RCU_DMA0);
	
	/* 复位DMA */
    dma_deinit(DMA0, DMA_CH0);
	
	/* 复位DMA参数结构体变量 */
	dma_struct_para_init(&dma_init_struct);
	
    /* 初始化各个参数 */
    dma_init_struct.periph_addr  = (uint32_t)(&ADC_RDATA(ADC0));
    dma_init_struct.periph_inc   = DMA_PERIPH_INCREASE_DISABLE;
    dma_init_struct.memory_addr  = (uint32_t)(&adc_value);
    dma_init_struct.memory_inc   = DMA_MEMORY_INCREASE_ENABLE;
    dma_init_struct.periph_width = DMA_PERIPHERAL_WIDTH_16BIT;		/* ADC精度为12位,此处必须大于12位 */
    dma_init_struct.memory_width = DMA_MEMORY_WIDTH_16BIT;			/* ADC精度为12位,此处必须大于12位 */
    dma_init_struct.direction    = DMA_PERIPHERAL_TO_MEMORY;
    dma_init_struct.number       = 2U;								/* 采集几路就设置为几 */
    dma_init_struct.priority     = DMA_PRIORITY_HIGH;
    dma_init(DMA0, DMA_CH0, &dma_init_struct);
	
	/* 使能循环模式 */
	dma_circulation_enable(DMA0, DMA_CH0);
	/* 不使用内存到内存模式 */
	dma_memory_to_memory_disable(DMA0, DMA_CH0);
	
    /* 使能DMA通道 */
    dma_channel_enable(DMA0, DMA_CH0);
}


注意:

①配置ADC的多个通道采样周期时,函数adc_regular_channel_config()的参数rank需要从0开始;

②DMA的外设及内存数据宽度需要大于ADC的精度12位;

③DMA的数据长度根据用户实际情况配置,可以采集几路配置为几,另外需要开启DMA的循环模式。

main.c文件中实现外设初始化及打印转换后的数据。

#include "gd32f10x.h"
#include "systick.h"
#include <stdio.h>
#include <string.h>
#include "main.h"
#include "bsp_usart.h"
#include "bsp_adc.h"

uint16_t	adc_value[2];

int main(void)
{
	float V[2];

	nvic_priority_group_set(NVIC_PRIGROUP_PRE2_SUB2);
    /* configure systick */
    systick_config();
	bsp_init_usart();
	bsp_init_adc();

    while(1)
	{		
		V[0] = adc_value[0] * 3.3 / 4095.0;            /* 计算电压值 */
		V[1] = adc_value[1] * 3.3 / 4095.0;

		printf("adc_value_0 = %d\r\n",adc_value[0]);    /* 打印转换后的12位AD码值 */
		printf("v_value_0 = %f\r\n",V[0]);              /* 打印计算得到的电压值 */
		printf("adc_value_1 = %d\r\n",adc_value[1]);
		printf("v_value_1 = %f\r\n",V[1]);

		delay_1ms(1000);
    }
}

/* retarget the C library printf function to the USART */
int fputc(int ch, FILE *f)
{
    usart_data_transmit(USART0, (uint8_t)ch);
    while(RESET == usart_flag_get(USART0, USART_FLAG_TBE));

    return ch;
}

串口打印结果如图。

其中,

adc_value_0是5V输入通过两个电阻分压后输入到单片机引脚PA1, 

adc_value_1是将3.3V直接输入到单片机引脚PA3。

PA1电压采集电路如下图。

由图可知,VBUS一般通过usb供电为5V,PA1引脚的电压即为2.5V,与串口打印数值吻合,说明此处ADC电压采集正确。

通过这种电阻分压的方式,就可以不受单片机引脚耐压3.3V的限制了,理论上就可以自由设置ADC的电压测量范围了。比如此处如果R43选为90K,R44选为10K,理论上就可以实现采集0-33V电压信号了,只需要在程序里加入相应的换算即可。但是,通过这种方式测量电压,需要注意选用精度较高的电阻,同时也可以加一颗旁路电容,减小测量误差。

  • 5
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: STM32F103是意法半导体(STMicroelectronics)公司推出的一款基于ARM Cortex-M3内核的32位微控制器。它具有丰富的外设和高性能,常用于各种应用领域。 其中的DMA(Direct Memory Access)是一种高速数据传输方式,它可以在不经过CPU的干预下,直接将数据从外设传输到内存,或者从内存传输到外设。这种方式可以显著提高数据传输的效率。STM32F103中的DMA控制器可以与多个外设进行数据传输,包括ADC和TIM。 ADC(Analog-to-Digital Converter)是模数转换器,用于将模拟信号转换为数字信号。STM32F103ADC模块具有多个通道,可以同时采集多个模拟信号,并将其转换为相应的数字值。通过DMA控制器,ADC模块可以将采集到的数据直接传输到内存,而无需CPU的干预。这样可以实现高效的模拟信号采集。 TIM(Timer)是定时器,可以用于生成各种定时、计数和PWM信号。STM32F103的定时器模块具有多个通道和多种功能,可以用于测量时间间隔、产生定时中断、实现PWM输出等。通过DMA控制器,定时器模块可以将产生的定时或PWM数据直接传输到外设,如LED驱动器等,大大减轻CPU的负担。 综上所述,STM32F103中的DMAADC和TIM模块具有协同工作的能力,可以实现高效的数据传输和信号采集。通过合理配置和使用这些模块,可以大大提升系统的性能和效率。 ### 回答2: STM32F103是一款基于ARM Cortex-M3内核的微控制器,具有丰富的外设资源。DMA(Direct Memory Access)是一种直接内存访问技术,可以实现外设与内存之间的数据传输,提高系统的数据传输效率。ADC(Analog-to-Digital Converter)是模数转换器,用于将模拟信号转换为数字信号。TIM(Timer)则是定时器,用于产生定时和计数操作。 STM32F103支持DMA控制器和多个ADC通道,这使得在数据采集过程中可以使用DMA来直接将ADC采样的数据传输到内存中,而无需CPU参与,从而提高了系统的效率。通过配置DMA通道和ADCDMA请求,可以实现自动转换和传输。在传输完成之后,可以通过DMA传输完成中断来进行相应的处理。 另外,STM32F103还具备多个定时器(TIM),其中包括基本定时器和通用定时器。可以使用定时器来产生特定的时间间隔,并触发相应的事件。通过配置定时器的预分频器、计数器和各种模式,可以满足不同的定时和计数需求。 综上所述,使用STM32F103DMAADC和TIM外设可以实现高效的数据采集和定时操作。通过合理配置和编程,可以满足不同应用场景下的实时数据采集和事件触发需求。 ### 回答3: STM32F103是意法半导体(STMicroelectronics)生产的一款32位单片机,它具有强大的性能和丰富的外设功能。其中DMA(Direct Memory Access,直接内存访问)是一种数据传输技术,ADC(Analog-to-Digital Converter,模数转换器)是一种模拟信号转换为数字信号的设备,而TIM(Timer/Counter,定时器/计数器)是用于计时和计数的设备。 在STM32F103中,DMA用于优化数据传输,通过在外设和存储器之间建立直接通道,实现数据的高速传输,减轻了单片机的处理负担。ADC是用于将模拟信号转换为数字信号的重要功能模块,可以采集外部传感器的模拟信号,并将其转换为数字形式供单片机处理。而TIM用来实现各种计时和计数功能,例如控制器模块中的PWM(Pulse Width Modulation,脉冲宽度调制)输出、测量时间间隔等。 在应用中,可以使用DMA来优化ADC的数据传输。通过配置DMA控制器,可以在ADC转换值就绪后,自动将转换结果复制到指定的存储位置,实现高效的数据传输。另外,TIM可以与ADC结合使用,通过定时触发ADC的转换,实现周期性采样。这样可以实现周期性信号的连续采样,满足一些实时性要求较高的应用场景。 总之,STM32F103DMAADC和TIM等功能模块的结合应用可以实现高效的数据传输和处理,满足各种应用需求。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值