目录
ADC的使用
ADC的初始化
void ADC_INIT(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_ADC1,ENABLE);
RCC_ADCCLKConfig( RCC_PCLK2_Div6 );
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init( GPIOB,&GPIO_InitStructure );//注意这里选择对PB引脚初始化
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 1; //比赛只需用一个通道
ADC_Init( ADC1,&ADC_InitStructure);
ADC_Cmd(ADC1,ENABLE);
ADC_ResetCalibration(ADC1);//复位
while(ADC_GetResetCalibrationStatus(ADC1));
ADC_StartCalibration(ADC1);//校准
while(ADC_GetCalibrationStatus(ADC1));
}
下面是数模转换模块的外围电路以及芯片,可以看到是对应的GPIOB口的Pin0引脚所以需要对RCC_APB2Periph_GPIOB进行时钟使能;这里使用的是ADC1,所以需要对ADC1进行使能(APB1与APB2与ADC的关系请参考:RCC_APB2PeriphClockCmd与镜像函数RCC_APB1PeriphClockCmd函数 https://blog.youkuaiyun.com/weixin_44599681/article/details/114853571):
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_ADC1,ENABLE);
但是由于芯片内部有ADC1、ADC2、ADC3,根据下面的表,我们可以看出ADC1和ADC2的前面0-15的通道的功能使用都一样的,所以根据上面的初始化代码,用ADC1还是ADC2都是一样的。
分频因子要确保 ADC1 的时钟(ADCCLK) 不要超过 14Mhz。 这个我们设置分频因子位 6,时钟为 72/6=12MHz,库函数的实现方法是:
RCC_ADCCLKConfig( RCC_PCLK2_Div6 );
在对GPIO的初始化结构体进行初始化时,需要注意的是使用的时ADC,且是信号的输入,所以GPIO_Mode用GPIO_Mode_AIN(模拟输入)
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;//模拟输入
以下代码是配置ADC的初始化结构体:
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfChannel = 1; //比赛只需用一个通道
ADC_Init( ADC1,&ADC_InitStructure);
对于STM32,在使用ADC的时候需要配置几个参数:
1.第一个参数是ADC_Mode,这里设置为独立模式:
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
在这个模式下,双ADC不能同步,每个ADC接口独立工作。所以如果不需要ADC同步或者只是用了一个ADC的时候,就应该设成独立模式了
2.第二个参数是ADC_ScanConvMode,这里设置为DISABLE。
ADC_InitStructure.ADC_ScanConvMode = DISABLE;
如果只是用了一个通道的话,DISABLE就可以了,如果使用了多个通道的话,则必须将其设置为ENABLE。
3.第三个参数是
ADC_ContinuousConvMode
,这里设置为ENABLE
,即连续转换。如果设置为DISABLE
,则是单次转换。两者的区别在于连续转换直到所有的数据转换完成后才停止转换,而单次转换则只转换一 次数据就停止,要再次触发转换才可以。所以如果需要一次性采集1024个数据或者更多,则采用连续转换。4.第四个参数是
ADC_ExternalTrigConv
,即选择外部触发模式。这里只讲三种:第一种是最简单的软件触发,参数为ADC_ExternalTrigConv_None。设置好后还要记得调用库函数:
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
后面的用法就不介绍了,详情请看引用结尾的连接。
5.第五个参数是
ADC_DataAlign
,这里设置为ADC_DataAlign_Righ
t右对齐方式。建议采用右对齐方式,因为这样处理数据会比较方便。当然如果要从高位开始传输数据,那么采用左对齐优势就明显了。6.第六个参数是
ADC_NbrOfChannel
,顾名思义:通道的数量。要是到多个通道采集数据的话就得设置一下这个参数。此外在规则通道组的配置函数中也许将各个通道的顺序定义一下,如:ADC_RegularChannelConfig(ADC1,ADC_Channel_13,1,ADC_SampleTime_13Cycles5); ADC_RegularChannelConfig(ADC1,ADC_Channel_14,2,ADC_SampleTime_13Cycles5);
多通道数据传输时有一点还要注意:若一个数组为
ADC_ValueTab[4]
,且设置了两个通道:通道1和通道2,则转换结束后,ADC_ValueTab[0]
和ADC_ValueTab[2]
存储的是通道1的数据,而ADC_ValueTab[1]
和ADC_ValueTab[3]
存储的是通道2的数据。如果数组容量大则依次类推详情请看:STM32之ADC配置,ADC_Mode模式理解 https://blog.youkuaiyun.com/qwq1503/article/details/96108617
ADC的一些基本控制方法
下面用以下函数来学习用ADC模块获取电压值对应的ADC的值:
u16 Get_ADC_Value(void)
{
u16 temp;
ADC_RegularChannelConfig(ADC1,ADC_Channel_8,1,ADC_SampleTime_239Cycles5);
ADC_SoftwareStartConvCmd(ADC1,ENABLE);
while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC) == 0);
temp = ADC_GetConversionValue(ADC1);
ADC_SoftwareStartConvCmd(ADC1,DISABLE);
return temp;
}
参考
[1] STM32之ADC配置,ADC_Mode模式理解 https://blog.youkuaiyun.com/qwq1503/article/details/96108617