基于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
}
}