GD32F450XX ADC2 DMA多路采集

因为GD官方给的例程中没有关于ADC2 DMA 独立模式多路采集的例程,所以在此上传一个。
直接上代码

#include "gd32f4xx.h"
#include "gd32f450i_eval.h"
#include "systick.h"
#include <stdio.h>

#define ADC_X ADC2

uint16_t adc_value[4];

void rcu_config(void);
void gpio_config(void);
void dma_config(void);
void adc_config(void);
void timer_config(void);

/*!
    \brief      main function
    \param[in]  none
    \param[out] none
    \retval     none
*/

int main(void)
{
    /* system clocks configuration */
    rcu_config();
    /* systick configuration */
    systick_config();  
    /* GPIO configuration */
    gpio_config();
	/* TIMER configuration */
    timer_config();

    /* ADC configuration */
    adc_config();
	/* DMA configuration */
    dma_config();
    /* configure COM port */
    gd_eval_com_init(EVAL_COM0);

    while(1){
        delay_1ms(1000);
        printf("\r\n *******************************");
        printf("\r\n ADC0 regular channel data = 0x%04X", adc_value[0]);
        printf("\r\n ADC0 regular channel data = 0x%04X", adc_value[1]);
        printf("\r\n ADC0 regular channel data = 0x%04X", adc_value[2]);
        printf("\r\n ADC0 regular channel data = 0x%04X\r\n", adc_value[3]);
    }
}

/*!
    \brief      configure the different system clocks
    \param[in]  none
    \param[out] none
    \retval     none
*/
void rcu_config(void)
{
    /* enable GPIOC clock */
    rcu_periph_clock_enable(RCU_GPIOA);
    /* enable ADC clock */
    rcu_periph_clock_enable(RCU_ADC0);
	rcu_periph_clock_enable(RCU_ADC2);
    /* enable timer1 clock */
    rcu_periph_clock_enable(RCU_TIMER1);
    rcu_timer_clock_prescaler_config(RCU_TIMER_PSC_MUL4);
    /* enable DMA clock */
    rcu_periph_clock_enable(RCU_DMA1);
    /* config ADC clock */
    adc_clock_config(ADC_ADCCK_PCLK2_DIV6);
}

/*!
    \brief      configure the GPIO peripheral
    \param[in]  none
    \param[out] none
    \retval     none
*/
void gpio_config(void)
{
    /* config the GPIO as analog mode */
    gpio_mode_set(GPIOA, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, GPIO_PIN_1);
    gpio_mode_set(GPIOA, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, GPIO_PIN_2);
    gpio_mode_set(GPIOA, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, GPIO_PIN_3);
    gpio_mode_set(GPIOA, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, GPIO_PIN_4 );
}

/*!
    \brief      configure the DMA peripheral
    \param[in]  none
    \param[out] none
    \retval     none
*/
void dma_config(void)
{
    /* ADC_DMA_channel configuration */
    dma_single_data_parameter_struct dma_single_data_parameter;
    
    /* ADC DMA_channel configuration */
    dma_deinit(DMA1, DMA_CH0);
    
    /* initialize DMA single data mode */
    dma_single_data_parameter.periph_addr = (uint32_t)(&ADC_RDATA(ADC_X));
    dma_single_data_parameter.periph_inc = DMA_PERIPH_INCREASE_DISABLE;
    dma_single_data_parameter.memory0_addr = (uint32_t)(adc_value);
    dma_single_data_parameter.memory_inc = DMA_MEMORY_INCREASE_ENABLE;
    dma_single_data_parameter.periph_memory_width = DMA_PERIPH_WIDTH_16BIT;
    dma_single_data_parameter.direction = DMA_PERIPH_TO_MEMORY;
    dma_single_data_parameter.number = 4;
    dma_single_data_parameter.priority = DMA_PRIORITY_HIGH;
    dma_single_data_mode_init(DMA1, DMA_CH0, &dma_single_data_parameter);
    dma_channel_subperipheral_select(DMA1, DMA_CH0, DMA_SUBPERI0);

    /* enable DMA circulation mode */
    dma_circulation_enable(DMA1, DMA_CH0);
	dma_channel_subperipheral_select(DMA1, DMA_CH0, DMA_SUBPERI2);
    /* enable DMA channel */
    dma_channel_enable(DMA1, DMA_CH0);
}

/*!
    \brief      configure the ADC peripheral
    \param[in]  none
    \param[out] none
    \retval     none
*/
void adc_config(void)
{
    /* ADC mode config */
    adc_sync_mode_config(ADC_SYNC_MODE_INDEPENDENT);
    /* ADC contineous function disable */
    adc_special_function_config(ADC_X, ADC_CONTINUOUS_MODE, DISABLE);
    /* ADC scan mode disable */
    adc_special_function_config(ADC_X, ADC_SCAN_MODE, ENABLE);
    /* ADC data alignment config */
    adc_data_alignment_config(ADC_X, ADC_DATAALIGN_RIGHT);
    
    /* ADC channel length config */
    adc_channel_length_config(ADC_X, ADC_REGULAR_CHANNEL, 4);
    /* ADC regular channel config */
    adc_regular_channel_config(ADC_X, 0, ADC_CHANNEL_1, ADC_SAMPLETIME_15);
    adc_regular_channel_config(ADC_X, 1, ADC_CHANNEL_2, ADC_SAMPLETIME_15);
    adc_regular_channel_config(ADC_X, 2, ADC_CHANNEL_3, ADC_SAMPLETIME_15);
    adc_regular_channel_config(ADC_X, 3, ADC_CHANNEL_4, ADC_SAMPLETIME_15);
    /* ADC trigger config */
    adc_external_trigger_source_config(ADC_X, ADC_REGULAR_CHANNEL, ADC_EXTTRIG_REGULAR_T1_CH1); 
    adc_external_trigger_config(ADC_X, ADC_REGULAR_CHANNEL, EXTERNAL_TRIGGER_RISING);
    
    /* ADC DMA function enable */
    adc_dma_request_after_last_enable(ADC_X);
    adc_dma_mode_enable(ADC_X);
    
    /* enable ADC interface */
    adc_enable(ADC_X);
    /* wait for ADC stability */
    delay_1ms(1);
    /* ADC calibration and reset calibration */
    adc_calibration_enable(ADC_X);
    
//    /* enable ADC software trigger */
//    adc_software_trigger_enable(ADC2, ADC_REGULAR_CHANNEL);
}
/*!
    \brief      configure the timer peripheral
    \param[in]  none
    \param[out] none
    \retval     none
*/
void timer_config(void)
{
    timer_oc_parameter_struct timer_ocintpara;
    timer_parameter_struct timer_initpara;

    /* TIMER1 configuration */
    timer_initpara.prescaler         = 19999;
    timer_initpara.alignedmode       = TIMER_COUNTER_EDGE;
    timer_initpara.counterdirection  = TIMER_COUNTER_UP;
    timer_initpara.period            = 9999;
    timer_initpara.clockdivision     = TIMER_CKDIV_DIV1;
    timer_initpara.repetitioncounter = 0;
    timer_init(TIMER1,&timer_initpara);

    /* CH1 configuration in PWM mode0 */
    timer_ocintpara.ocpolarity  = TIMER_OC_POLARITY_HIGH;
    timer_ocintpara.outputstate = TIMER_CCX_ENABLE;
    timer_channel_output_config(TIMER1,TIMER_CH_1,&timer_ocintpara);

    timer_channel_output_pulse_value_config(TIMER1,TIMER_CH_1,9900);
    timer_channel_output_mode_config(TIMER1,TIMER_CH_1,TIMER_OC_MODE_PWM0);
    timer_channel_output_shadow_config(TIMER1,TIMER_CH_1,TIMER_OC_SHADOW_DISABLE);
    
    /* enable TIMER1 */
    timer_enable(TIMER1);
}
/* retarget the C library printf function to the USART */
int fputc(int ch, FILE *f)
{
    usart_data_transmit(EVAL_COM0, (uint8_t)ch);
    while(RESET == usart_flag_get(EVAL_COM0, USART_FLAG_TBE));
    return ch;
}

1,首先第一个需要注意的地方是 官方的例程中 关于DMA的配置
在这里插入图片描述
官方使用的是 DMA1 的通道0 ADC0
dma_channel_subperipheral_select(DMA1, DMA_CH0, DMA_SUBPERI0);
而我们要配置为 DMA1 的通道0 ADC2
dma_channel_subperipheral_select(DMA1, DMA_CH0, DMA_SUBPERI2);

2,如果要使用软件触发的话需要同时 使能扫描模式和连续转换模式,如果只使能扫描模式的话只会进行一次转换。
/* ADC contineous function disable /
adc_special_function_config(ADC_X, ADC_CONTINUOUS_MODE, ENABLE);
/
ADC scan mode disable */
adc_special_function_config(ADC_X, ADC_SCAN_MODE, ENABLE);

以上问题也是问了代理商技术支持后解决的,还是感谢技术支持,大家共勉,避免踩坑。

  • 1
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
嗨!对于使用GD32F450微控制器的ADC DMA,您可以按照以下步骤进行配置: 1. 首先,确保您已经初始化了ADCDMA外设,并启用了它们的时钟。 2. 配置ADC的通道和采样时间: - 使用`adc_channel_length`函数设置ADC的通道数。 - 使用`adc_regular_channel_config`函数配置每个通道的采样时间、转换模式和触发源。 3. 配置DMA: - 使用`dma_deinit`函数重置DMA外设。 - 使用`dma_init_struct`结构体配置DMA的通道、数据传输方向、数据宽度和传输模式。 - 使用`dma_mode_config`函数配置DMA的传输模式。 - 使用`dma_circulation_disable`函数禁用DMA的循环传输模式(如果不需要循环传输)。 - 使用`dma_memory_address_config`函数配置DMA的内存地址和外设地址。 - 使用`dma_memory_width_config`函数配置DMA的内存和外设数据宽度。 - 使用`dma_memory_increase_enable`函数启用DMA的内存递增模式。 - 使用`dma_channel_enable`函数启用DMA通道。 4. 启动ADCDMA: - 使用`adc_software_trigger_enable`函数启动ADC转换。 - 使用`dma_channel_enable`函数启动DMA传输。 5. 在DMA传输完成后,您可以使用DMA的中断或轮询方法来处理接收到的数据。 这是一个基本的ADC DMA配置过程的概述。具体的代码实现可能会有所不同,具体取决于您使用的开发环境和库。确保参考GD32F450微控制器的参考手册和相关的库文档以获取更详细的信息和示例代码。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值