ADC独立模式-多通道采集实验

在stm32F中,ADC1/3具有DMA请求,ADC2 不具有。
在这里插入图片描述
在这里插入图片描述

这里多通道(6)的实现,以ADC1来实现,创建bsp_adc.c和bsp_adc.h两个文件:
bsp_adc.h

#ifndef __BSP_ADC_H_
#define __BSP_ADC_H_

#include "stm32f10x.h"

// ADC GPIO宏定义
// 注意:用作ADC采集的IO必须没有复用,否则采集电压会有影响
#define    ADC_GPIO_APBxClock_FUN        RCC_APB2PeriphClockCmd
#define    ADC_GPIO_CLK                  RCC_APB2Periph_GPIOC  
#define    ADC_PORT                      GPIOC
// 多通道宏定义   转换通道个数6
#define    NOFCHANEL				     6  

#define    ADC_PIN1                      GPIO_Pin_0
#define    ADC_CHANNEL1                  ADC_Channel_10

#define    ADC_PIN2                      GPIO_Pin_1
#define    ADC_CHANNEL2                  ADC_Channel_11

#define    ADC_PIN3                      GPIO_Pin_2
#define    ADC_CHANNEL3                  ADC_Channel_12

#define    ADC_PIN4                      GPIO_Pin_3
#define    ADC_CHANNEL4                  ADC_Channel_13

#define    ADC_PIN5                      GPIO_Pin_4
#define    ADC_CHANNEL5                  ADC_Channel_14

#define    ADC_PIN6                      GPIO_Pin_5
#define    ADC_CHANNEL6                  ADC_Channel_15

// ADC 编号选择,只有ADC1/3具有DMA请求,所以可以选则ADC1或者3
#define    ADC_APBxClock_FUN             RCC_APB2PeriphClockCmd
#define    ADC_x                         ADC1
#define    ADC_CLK                       RCC_APB2Periph_ADC1

// ADC 通道宏定义
#define   ADC_CHANNEL                   ADC_Channel_11
//ADC1和3之间的DMA请求通过宏定义来进行修改
/*
//ADC3的DMA请求宏定义,根据前面的手册图片可以知道
#define   ADC_DMA_CLK                   RCC_AHBPeriph_DMA2
#define   ADC_DMA_CHANNEL               DMA2_Channel5
*/
#define   ADC_DMA_CLK                   RCC_AHBPeriph_DMA1
#define   ADC_DMA_CHANNEL               DMA1_Channel1

void ADCx_Init(void);

#endif

bsp_adc.c

#include "bsp_adc.h"

__IO uint16_t ADC_ConvertedValue[NOFCHANEL];    //6通道的,用数组来定义

static void ADCx_GPIO_Config(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	// 打开 ADC IO端口时钟
	ADC_GPIO_APBxClock_FUN ( ADC_GPIO_CLK, ENABLE );
	
	// 配置 ADC IO 引脚模式
	// 必须为模拟输入
	GPIO_InitStructure.GPIO_Pin = 	ADC_PIN1|
						            ADC_PIN2|
									ADC_PIN3|
									ADC_PIN4|
									ADC_PIN5|
									ADC_PIN6;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	
	// 初始化 ADC IO
	GPIO_Init(ADC_PORT, &GPIO_InitStructure);		
}

static void ADCx_Mode_Config(void)
{
     DMA_InitTypeDef DMA_InitStructure;
	
	// 打开DMA时钟
	RCC_AHBPeriphClockCmd(ADC_DMA_CLK, ENABLE);
	// 复位DMA控制器
	DMA_DeInit(ADC_DMA_CHANNEL);
	
	// 配置 DMA 初始化结构体
	// 外设基址为:ADC 数据寄存器地址
	DMA_InitStructure.DMA_PeripheralBaseAddr = ( uint32_t ) ( & ( ADC_x->DR ) );
	// 存储器地址,实际上就是一个内部SRAM的变量
	DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue;
	// 数据源来自外设
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
	// 缓冲区大小为6,缓冲区的大小应该等于存储器的大小
	DMA_InitStructure.DMA_BufferSize = NOFCHANEL;                       //改
	// 外设寄存器只有一个,地址不用递增
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
	// 存储器地址固定
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;             //改
	// 外设数据大小为半字,即两个字节
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
	// 存储器数据大小也为半字,跟外设数据大小相同
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
	// 循环传输模式
	DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;	
  // DMA 传输通道优先级为高,当使用一个DMA通道时,优先级设置不影响
	DMA_InitStructure.DMA_Priority = DMA_Priority_High;
	// 禁止存储器到存储器模式,因为是从外设到存储器
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
	// 初始化DMA
	DMA_Init(ADC_DMA_CHANNEL, &DMA_InitStructure);
	// 使能 DMA 通道
	DMA_Cmd(ADC_DMA_CHANNEL , ENABLE);	
	
/*................................................*/
	
	ADC_InitTypeDef ADC_InitStructure;
	
    // 打开ADC时钟
	ADC_APBxClock_FUN ( ADC_CLK, ENABLE );
	
	//ADC模式配置
	//只是用一个ADC,属于独立模式
	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;
	//转换通道6个
	ADC_InitStructure.ADC_NbrOfChannel = NOFCHANEL;
	//初始化ADC
	ADC_Init(ADC_x, &ADC_InitStructure);
	
	//配置ADC时钟为CLK的8分频,即9MHz
	RCC_ADCCLKConfig(RCC_PCLK2_Div8);
	// 配置ADC 通道的转换顺序(这里的转换顺序可以随意更改)和采样时间
	ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL1, 1, ADC_SampleTime_55Cycles5);
	ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL2, 2, ADC_SampleTime_55Cycles5);
	ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL3, 3, ADC_SampleTime_55Cycles5);
	ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL4, 4, ADC_SampleTime_55Cycles5);
	ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL5, 5, ADC_SampleTime_55Cycles5);
	ADC_RegularChannelConfig(ADC_x, ADC_CHANNEL6, 6, ADC_SampleTime_55Cycles5);

     //使能ADC_DMA请求
	ADC_DMACmd(ADC_x,ENABLE);
	//开启ADC,并开始转换
	ADC_Cmd(ADC_x,ENABLE);
	
	// 初始化ADC 校准寄存器  
	ADC_ResetCalibration(ADC_x);
	// 等待校准寄存器初始化完成
	while(ADC_GetResetCalibrationStatus(ADC_x));
	
	// ADC开始校准
	ADC_StartCalibration(ADC_x);
	// 等待校准完成
	while(ADC_GetCalibrationStatus(ADC_x));
	
	//由于没有采用外部触发,所以使用软件触发ADC转换
	ADC_SoftwareStartConvCmd(ADC_x,ENABLE);
}

/**
  * @brief  ADC初始化
  * @param  无
  * @retval 无
  */
void ADCx_Init(void)
{
	ADCx_GPIO_Config();
	ADCx_Mode_Config();
}

main()

#include "stm32f10x.h"    //相当于51单片中的#include <reg51.h>
#include "bsp_led.h"
#include "bsp_usart.h" 
#include "bsp_adc.h" 

extern __IO uint16_t ADC_ConvertedValue[NOFCHANEL];

// 局部变量,用于保存转换计算后的电压值 	 
float ADC_ConvertedValueLocal[NOFCHANEL];   

// 软件延时
void Delay(__IO uint32_t nCount)
{
  for(; nCount != 0; nCount--);
} 

int main(void)
{
   //配置串口
	 USART_Config(); 
	 //ADC初始化
	 ADCx_Init();
	 printf("\r\n ----这是一个ADC单通道中断读取实验----\r\n");
	 while(1)
	{
	   
			ADC_ConvertedValueLocal[0] =(float) ADC_ConvertedValue[0]/4096*3.3;
			ADC_ConvertedValueLocal[1] =(float) ADC_ConvertedValue[1]/4096*3.3;
			ADC_ConvertedValueLocal[2] =(float) ADC_ConvertedValue[2]/4096*3.3;
			ADC_ConvertedValueLocal[3] =(float) ADC_ConvertedValue[3]/4096*3.3;
			ADC_ConvertedValueLocal[4] =(float) ADC_ConvertedValue[4]/4096*3.3;
			ADC_ConvertedValueLocal[5] =(float) ADC_ConvertedValue[5]/4096*3.3;
		    ADC_ConvertedValueLocal[6] =(float) ADC_ConvertedValue[6]/4096*3.3;
		
			printf("\r\n CH0 value = %f V \r\n",ADC_ConvertedValueLocal[0]);
			printf("\r\n CH1 value = %f V \r\n",ADC_ConvertedValueLocal[1]);
			printf("\r\n CH2 value = %f V \r\n",ADC_ConvertedValueLocal[2]);
			printf("\r\n CH3 value = %f V \r\n",ADC_ConvertedValueLocal[3]);
			printf("\r\n CH4 value = %f V \r\n",ADC_ConvertedValueLocal[4]);
			printf("\r\n CH5 value = %f V \r\n",ADC_ConvertedValueLocal[5]);
		    printf("\r\n CH6 value = %f V \r\n",ADC_ConvertedValueLocal[6]);
		
		    printf("\r\n\r\n");
		    Delay(0XFFFFEE);
	}
}
  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是GD32常规模式下的多通道ADC采集中断的基本步骤: 1. 配置ADC模块,并开启DMA传输。在多通道模式下,需要设置扫描模式,以及通道顺序和数量。 2. 配置NVIC中断,并使能中断。在多通道模式下,需要使能每个通道的转换完成中断。 3. 在ADC中断处理函数中,获取转换结果,并更新结果数组。 下面是一个简单的示例代码: ```c #include "gd32f1x0.h" #include <stdio.h> #define ADC_GPIO_PORT GPIOB #define ADC_GPIO_PIN GPIO_PIN_0 | GPIO_PIN_1 | GPIO_PIN_2 #define ADC_GPIO_PORT_CLK RCU_GPIOB #define ADC_DMA_CHANNEL DMA_CH0 #define ADC_DMA_CHANNEL_CLK RCU_DMA0 #define ADC_MODULE ADC0 #define ADC_MODULE_CLK RCU_ADC0 #define ADC_SAMPLE_COUNT 3 #define ADC_INT_PRIO 2 uint16_t adc_results[ADC_SAMPLE_COUNT]; void adc_init(void) { /* enable GPIO clock */ rcu_periph_clock_enable(ADC_GPIO_PORT_CLK); /* enable ADC clock */ rcu_periph_clock_enable(ADC_MODULE_CLK); /* enable DMA clock */ rcu_periph_clock_enable(ADC_DMA_CHANNEL_CLK); /* configure GPIO pins as analog inputs */ gpio_mode_set(ADC_GPIO_PORT, GPIO_MODE_ANALOG, GPIO_PUPD_NONE, ADC_GPIO_PIN); /* configure ADC */ adc_deinit(ADC_MODULE); adc_mode_config(ADC_MODE_FREE); adc_special_function_config(ADC_SCAN_MODE, ENABLE); adc_data_alignment_config(ADC_DATAALIGN_RIGHT); adc_channel_length_config(ADC_SAMPLE_COUNT); adc_regular_channel_config(0, ADC_CHANNEL_10, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(1, ADC_CHANNEL_11, ADC_SAMPLETIME_55POINT5); adc_regular_channel_config(2, ADC_CHANNEL_12, ADC_SAMPLETIME_55POINT5); adc_external_trigger_source_config(ADC_INSERTED_CHANNEL, ADC_EXTERNAL_TRIGGER_SOURCE_NONE); adc_dma_mode_enable(); adc_interrupt_enable(ADC_INT_SCAN); adc_enable(); } void dma_init(void) { /* configure DMA channel */ dma_deinit(ADC_DMA_CHANNEL); dma_channel_select(ADC_DMA_CHANNEL, DMA_REQUEST_ADC0_1); dma_transfer_direction_config(ADC_DMA_CHANNEL, DMA_PERIPHERAL_TO_MEMORY); dma_memory_address_config(ADC_DMA_CHANNEL, (uint32_t)adc_results); dma_memory_width_config(ADC_DMA_CHANNEL, DMA_MEMORY_WIDTH_16BIT); dma_periph_width_config(ADC_DMA_CHANNEL, DMA_PERIPHERAL_WIDTH_16BIT); dma_priority_config(ADC_DMA_CHANNEL, DMA_PRIORITY_HIGH); dma_transfer_number_config(ADC_DMA_CHANNEL, ADC_SAMPLE_COUNT); dma_periph_inc_mode_config(ADC_DMA_CHANNEL, DMA_PERIPH_INCREASE_DISABLE); dma_memory_inc_mode_config(ADC_DMA_CHANNEL, DMA_MEMORY_INCREASE_ENABLE); dma_circulation_disable(ADC_DMA_CHANNEL); dma_memory_to_memory_disable(ADC_DMA_CHANNEL); dma_enable(ADC_DMA_CHANNEL); } void nvic_init(void) { /* configure ADC interrupt */ nvic_irq_enable(ADC0_1_IRQn, ADC_INT_PRIO, 0); } void ADC0_1_IRQHandler(void) { if (adc_interrupt_flag_get(ADC_MODULE, ADC_INT_FLAG_EOCS)) { adc_interrupt_flag_clear(ADC_MODULE, ADC_INT_FLAG_EOCS); dma_channel_disable(ADC_DMA_CHANNEL); } } int main(void) { adc_init(); dma_init(); nvic_init(); /* start ADC conversion */ adc_software_trigger_enable(ADC_MODULE, ADC_INSERTED_CHANNEL); /* wait for DMA transfer to complete */ while (dma_channel_enabled(ADC_DMA_CHANNEL)); /* print ADC results */ printf("ADC results: %d, %d, %d\n", adc_results[0], adc_results[1], adc_results[2]); while (1); } ``` 在该示例代码中,ADC模块被配置为扫描三个通道,并通过DMA传输转换结果。在ADC中断处理函数中,当所有通道的转换完成后,DMA传输被禁用,并且转换结果被保存在结果数组中。最后,主循环中打印转换结果。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值