IN0~IN7,通过通道选择开关,选中一路,输入到比较器进行转换和比较。首先是一个电压比较器,它可以判断两个输入信号电压的大小关系,输出一个高低电平指示谁大谁大小。 这个比较器有两个输入端,一个是待测电压,另一个是DAC电压输出端,DAC是数模转换器,给它一个数据,他就能输出数据对应的电压。一个是外部输入端的未知编码电压,一个是DAC输出的已知编码电压,他们同时输入电压比较器,进行大小判断。如果DAC输出的电压比较大,就调小DAC数据,如果DAC输出电压小,就增大DAC数据,直到DAC输出的电压和外部通道输入的电压近似相等,那么DAC输入的数据就是外部电压的编码数据。
使用规则通道配合DMA就可以不用担心数据被覆盖。
EOC是规则组的完成信号,JEOC是注入组的完成信号,这两个信号都会再状态寄存器里置一个标志位,读取标志位就可以知道是否转换结束。
比如序列1为通道2,之后就可以触发转换,ADC对通道2进行模数转换,过一定的时间后,转换完成,转换结果放在数据寄存器里,同时给EOC标志位置1。
连续转换与单次转换不同的是,它再一次转换结束后不会停止,而是立刻开始下一轮的转换,之后一直持续下去。
单次转换跟上面的单次转换非扫描模式大致一样,但是扫描模式就会用到菜单列表,可以连续对选中的位置进行转换,转换结果都放在数据寄存器里,但为了防止数据被覆盖,需要用到DMA及时把数据挪走,7个通道转换完成之后,产生EOC信号,转换结束。
12位ADC,但数据寄存器有16位,就分为数据右对齐和左对齐。一般情况下是选择数据右对齐,数据左对齐会让误差变大,因为一位就是要乘2,进4位就是乘16,所以一般不用,除非就用简单的判断,把数据的高8位取出来,舍弃后4位精度。
#include "Bsp_ADC.h"
void Bsp_ADC_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); // 1.时钟配置
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 2.ADC分频,ADC最大14M,接近来的时候是72M,所以要分频到14M以下
GPIO_InitTypeDef GPIO_InitStructure; // 3.GPIO配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); // 4.配置规则组
// ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 2, ADC_SampleTime_55Cycles5); // 相当于填充菜单列表方法
ADC_InitTypeDef ADC_InitStructure; // 5.配置ADC
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // 数据对齐
ADC_InitStructure.ADC_NbrOfChannel = 1;
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; // 外部触发转换选择
//ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; // 连续转换模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE); // 6.开启ADC
ADC_ResetCalibration(ADC1); // 7.校准ADC
while (ADC_GetResetCalibrationStatus(ADC1) == SET);
ADC_StartCalibration(ADC1);
while (ADC_GetCalibrationStatus(ADC1) == SET);
ADC_SoftwareStartConvCmd(ADC1, ENABLE); // ADC软件启动转换,因为开启了连续转换模式,所以只开启一次即可
// 如果需要看门狗和中断,则需要额外配置(开启看门狗和中断配置)
}
uint16_t ADC_GetValue(void)
{
//ADC_SoftwareStartConvCmd(ADC1, ENABLE); // ADC软件启动转换
//while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); // 等待ADC读取EOC标志位为1; 并且也不需要判断标志位了
return ADC_GetConversionValue(ADC1); // 返回ADC1的值
}
#include "Bsp_ADC.h"
void Bsp_ADC_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); // 1.时钟配置
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 2.ADC分频,ADC最大14M,接近来的时候是72M,所以要分频到14M以下
GPIO_InitTypeDef GPIO_InitStructure; // 3.GPIO配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
// ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); // 4.配置规则组
// ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 2, ADC_SampleTime_55Cycles5); // 相当于填充菜单列表方法
ADC_InitTypeDef ADC_InitStructure; // 5.配置ADC
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // 数据对齐
ADC_InitStructure.ADC_NbrOfChannel = 1;
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; // 外部触发转换选择
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; // 连续转换模式
// ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE); // 6.开启ADC
ADC_ResetCalibration(ADC1); // 7.校准ADC
while (ADC_GetResetCalibrationStatus(ADC1) == SET);
ADC_StartCalibration(ADC1);
while (ADC_GetCalibrationStatus(ADC1) == SET);
// ADC_SoftwareStartConvCmd(ADC1, ENABLE); // ADC软件启动转换,因为开启了连续转换模式,所以只开启一次即可
// 如果需要看门狗和中断,则需要额外配置(开启看门狗和中断配置)
}
/* 读取ADC的每个通道,在主程序里读取4次即可完成多通道 */
uint16_t ADC_GetValue(uint8_t ADC_Channel)
{
ADC_SoftwareStartConvCmd(ADC1, ENABLE); // ADC软件启动转换
while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); // 等待ADC读取EOC标志位为1; 并且也不需要判断标志位了
ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5);
return ADC_GetConversionValue(ADC1); // 返回ADC1的值
}