【GD32】基于GD32F303CCT6的AD采样-单通道+DMA+软件触发+连续模式

GD32官方提供的手册并没有标注各个通道对应的引脚,在开发的时候参考的STM32的通道引脚对照表没想到竟然可以照用不误。
在这里插入图片描述

这里选择的是使用PA7-AD0-通道7,下面是运行的代码(实测可行)。

一、初始化GPIO和RCU时钟

void static ADC_GPIOAndRCU_Config(void)
{
    //GPIO设置为模拟输入即可
	gpio_init(GPIOA,GPIO_MODE_AIN,GPIO_OSPEED_MAX,GPIO_PIN_7);
	
	
	/* enable GPIOA clock */
	rcu_periph_clock_enable(RCU_GPIOA);
	/* enable ADC clock */
	rcu_periph_clock_enable(RCU_ADC0);
	/* enable DMA0 clock */
	rcu_periph_clock_enable(RCU_DMA0);//此处DMA用于移动ADC数据寄存器中的数据到内存变量中
	/* config ADC clock */
	rcu_adc_clock_config(RCU_CKADC_CKAPB2_DIV4);
	
}

二、初始化DMA

void static ADC_DMA_Config(void)
{
	dma_parameter_struct dma_data_parameter;
	
	dma_deinit(DMA0,DMA_CH0);
	
	/* initialize DMA data mode */
	dma_data_parameter.periph_addr = (uint32_t)(&ADC_RDATA(ADC0));//这里选择ADC0的数据寄存器地址
	dma_data_parameter.periph_inc = DMA_PERIPH_INCREASE_DISABLE;
    
	dma_data_parameter.memory_addr = (uint32_t)(&adc_value);//这里选择自己定义的变量
	dma_data_parameter.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    
	dma_data_parameter.periph_width = DMA_PERIPHERAL_WIDTH_16BIT;//12位数据寄存器
	dma_data_parameter.memory_width = DMA_MEMORY_WIDTH_16BIT;  
    
	dma_data_parameter.direction = DMA_PERIPHERAL_TO_MEMORY;
	dma_data_parameter.number = 1;
	dma_data_parameter.priority = DMA_PRIORITY_HIGH;
	dma_init(DMA0, DMA_CH0, &dma_data_parameter);

	dma_circulation_enable(DMA0, DMA_CH0);

	/* enable DMA channel */
	dma_channel_enable(DMA0, DMA_CH0);//启动DMA
}

三、初始化ADC0

void static ADC_DMA_Config(void)
{	
	//********************************************************//
	/* ADC continous function enable */
    //由于是单通道所以关闭扫描模式
	adc_special_function_config(ADC0, ADC_SCAN_MODE, DISABLE);
    //启动连续模式进行数据转换
	adc_special_function_config(ADC0, ADC_CONTINUOUS_MODE, ENABLE);  
	
	/* ADC trigger config */
    //设置触发模式为规则通道软件触发
	adc_external_trigger_source_config(ADC0, ADC_REGULAR_CHANNEL, ADC0_1_2_EXTTRIG_REGULAR_NONE);

	/* ADC data alignment config */
    //采用右对齐
	adc_data_alignment_config(ADC0, ADC_DATAALIGN_RIGHT);

	/* ADC mode config */
    //选择单通道模式
	adc_mode_config(ADC_MODE_FREE); 
	/* ADC channel length config */
    //配置通道数目为1
	adc_channel_length_config(ADC0, ADC_REGULAR_CHANNEL, 1);
	
	/* ADC DMA function enable */
    //开启DMA模式
	adc_dma_mode_enable(ADC0);

	/* ADC regular channel config */
    //配置规则通道为来自那个引脚(见上方通道引脚对照表)
	adc_regular_channel_config(ADC0, 0, ADC_CHANNEL_8, ADC_SAMPLETIME_55POINT5);


	/* ADC external trigger enable */
    //外部软件触发使能
	adc_external_trigger_config(ADC0, ADC_REGULAR_CHANNEL, ENABLE);
  

	/* enable ADC interface */
	adc_enable(ADC0);  
    //等待ADC0稳定
	delay_1ms(1);
	/* ADC calibration and reset calibration */
	adc_calibration_enable(ADC0);
	
    //开启软件触发开始进行AD采样转换
	adc_software_trigger_enable(ADC0,ADC_REGULAR_CHANNEL);
}

不知道为什么老出现0.2v的零偏
效果如下图所示
在这里插入图片描述

四、使用的外部函数

#include "gd32f30x.h"
#include "systick.h"

volatile static uint32_t delay;

/*!
    \brief      configure systick
    \param[in]  none
    \param[out] none
    \retval     none
*/
void systick_config(void)
{
    /* setup systick timer for 1000Hz interrupts */
    if (SysTick_Config(SystemCoreClock / 1000U)){
        /* capture error */
        while (1){
        }
    }
    /* configure the systick handler priority */
    NVIC_SetPriority(SysTick_IRQn, 0x00U);
}

/*!
    \brief      delay a time in milliseconds
    \param[in]  count: count in milliseconds
    \param[out] none
    \retval     none
*/
void delay_1ms(uint32_t count)
{
    delay = count;

    while(0U != delay){
    }
}

/*!
    \brief      delay decrement
    \param[in]  none
    \param[out] none
    \retval     none
*/
void delay_decrement(void)
{
    if (0U != delay){
        delay--;
    }
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值