基于STM32F103C8T6双通道ADC转换非DMA数据存取

        基于STM32的双通道ADC采样非DMA数据存取。示例以GPIOA_PIN0,GPIOA_PIN1为模拟量输入引脚,对应ADC1的通道0,通道1。

       要实现非DMA数据存取,而能够实时读取不同通道的ADC转换值,需要将ADC转换模式设置为单次转换及非扫描模式,数据读取时规则组通道配置中将通道0,通道1分别配置到规则组序列1位置

       以下为全部代码(Keil5编写)

ADC.C

#include "stm32f10x.h"                  

/*AD初始化 选择GPIOA_PIN0/PIN1为模拟量输入引脚对应ADC1通道0/通道1*/
void AD_Init(void)
{
	/*开启GPIOA时钟*/
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); 
		
	/*GPIO初始化  将PA0/PA1引脚初始化为模拟输入*/
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);				
	
    /*开启ADC1时钟*/
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);  
   
    /*设置ADC时钟  选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz*/
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);	
	
	/**ADC初始化
      *定义结构体变量
      *模式选择,选择独立模式,即单独使用ADC1
      *数据对齐,选择右对齐
      *使用软件触发,不需要外部触发
      *连续转换,失能,每转换一次规则组序列后停止(多通道非DMA数据传输选择单次转换)
      *扫描模式,失能,只转换规则组的序列1这一个位置(多通道非DMA数据传输选择非扫描模式)
      *通道数,为1,仅在扫描模式下,才需要指定大于1的数,在非扫描模式下,只能是1
      *将结构体变量交给ADC_Init,配置ADC1
      */
	ADC_InitTypeDef ADC_InitStructure;						
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;		
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;	
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;	
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;		
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;			
	ADC_InitStructure.ADC_NbrOfChannel = 1;					
	ADC_Init(ADC1, &ADC_InitStructure);						
	
	/*ADC使能*/
	ADC_Cmd(ADC1, ENABLE);									
	
	/*ADC校准*/
	ADC_ResetCalibration(ADC1);								
	while (ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1) == SET);
}

/**AD数据采样  返回对应通道AD转换的值,范围:0~4095
  *规则组通道配置(规则组序列1的位置,分别配置为通道0,通道1)
  *软件触发对应通道AD转换一次
  *等待EOC标志位
  *读取数据寄存器,并返回对应通道AD转换的结果
  */

/*通道0采样*/
uint16_t AD1_GetValue(void)
{
	ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);		
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);					
	while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);	
	return ADC_GetConversionValue(ADC1);					
}

/*通道1采样*/
uint16_t AD2_GetValue(void)
{
	ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_55Cycles5);		
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);					
	while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);	
	return ADC_GetConversionValue(ADC1);					
}

ADC.H

#ifndef __ADC_H
#define __ADC_H

void AD_Init(void);
uint16_t AD1_GetValue(void);
uint16_t AD2_GetValue(void);

#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "ADC.h"


/*变量名定义*/
uint16_t ADValue1;		//定义通道0转换值的变量
uint16_t ADValue2;		//定义通道1转换值的变量
float Voltage1;			//定义通道0即AD1电压变量
float Voltage2;			//定义通道1即AD2电压变量

int main(void)
{
	/*模块初始化*/
	OLED_Init();		//OLED初始化
	AD_Init();			//AD初始化
	
	/*显示静态字符串*/
	OLED_ShowString(1, 1,  "AD1:0.00V");	    //1行1列显示字符串AD1:0.00V
	OLED_ShowString(2, 1,  "AD2:0.00V");		//2行1列显示字符串AD2:0.00V
	
	while (1)
	{	
        /*各通道AD数据读取及显示(模拟量输入0-3.3V 对应 0-4095)*/
		ADValue1 = AD1_GetValue();					            //获取通道0转换的值
		Voltage1 = (float)ADValue1 / 4095 * 3.3;		        //转换到0~3.3,表示AD1电压
		OLED_ShowNum(1, 5, Voltage1, 1);				        //显示AD1整数部分
		OLED_ShowNum(1, 7, (uint16_t)(Voltage1 * 100) % 100, 2);//显示AD1小数部分
		Delay_ms(100);			                                //延时100ms
		
		ADValue2 = AD2_GetValue();					            //获取通道1转换的值
		Voltage2 = (float)ADValue2 / 4095 * 3.3;		        //转换到0~3.3,表示AD2电压
		OLED_ShowNum(2, 5, Voltage2, 1);				        //显示AD2整数部分
		OLED_ShowNum(2, 7, (uint16_t)(Voltage2 * 100) % 100, 2);//显示AD2小数部分
		Delay_ms(100);			                                //延时100ms
		
	}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值