上代码:
先简单写一个ADC驱动步骤:
1.开ADC1和GPIO的时钟
2.设置ADC时钟分频
3.GPIO初始化
4.ADC初始化
5.ADC使能
6.AADC校准
-----------------------------
然后获取ADC的转换值
#include "stm32f10x.h" // Device header
#include "ad.h"
/**
* 函 数:AD初始化
* 参 数:无
* 返 回 值:无
*/
void AD_Init(void)
{
/*开启时钟*/
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //开启ADC1的时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //开启GPIOA的时钟
/*设置ADC时钟*/
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz
/*GPIO初始化*/
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure); //将PA0、PA1、PA2和PA3引脚初始化为模拟输入
/*不在此处配置规则组序列,而是在每次AD转换前配置,这样可以灵活更改AD转换的通道*/
/*ADC初始化*/
ADC_InitTypeDef ADC_InitStructure; //定义结构体变量
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //模式,选择独立模式,即单独使用ADC1
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //数据对齐,选择右对齐
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //外部触发,使用软件触发,不需要外部触发
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //连续转换,失能,每转换一次规则组序列后停止
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //扫描模式,失能,只转换规则组的序列1这一个位置
ADC_InitStructure.ADC_NbrOfChannel = 1; //通道数,为1,仅在扫描模式下,才需要指定大于1的数,在非扫描模式下,只能是1
ADC_Init(ADC1, &ADC_InitStructure); //将结构体变量交给ADC_Init,配置ADC1
/*ADC使能*/
ADC_Cmd(ADC1, ENABLE); //使能ADC1,ADC开始运行
/*ADC校准*/
ADC_ResetCalibration(ADC1); //固定流程,内部有电路会自动执行校准
while (ADC_GetResetCalibrationStatus(ADC1) == SET);
ADC_StartCalibration(ADC1);
while (ADC_GetCalibrationStatus(ADC1) == SET);
}
/**
* 函 数:获取AD转换的值
* 参 数:ADC_Channel 指定AD转换的通道,范围:ADC_Channel_x,其中x可以是0/1/2/3
* 返 回 值:AD转换的值,范围:0~4095
*/
uint16_t AD_GetValue(uint8_t ADC_Channel)
{
ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5); //在每次转换前,根据函数形参灵活更改规则组的通道1
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //软件触发AD转换一次
while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); //等待EOC标志位,即等待AD转换结束
return ADC_GetConversionValue(ADC1); //读数据寄存器,得到AD转换的结果
}
adc.h
#ifndef __AD_H
#define __AD_H
void AD_Init(void);
uint16_t AD_GetValue(uint8_t ADC_Channel);
#endif
主函数
/*-----------------------------------------------------------------------------------------------------------
*函数名称: Task_SynchEventHandle(void *pdata)
*功能描述: --任务同步事件处理函数
*输 入: *pdata 任务运行参数
*输 出: None
*返 回 值: None
*作 者: David Wang
*创建日期: 2015-03-10
*------------------------------------------------------------------------------------------------------------
*修 改 人:
*修改日期:
*修改描述:
-----------------------------------------------------------------------------------------------------------*/
void ADC_SynchEventHandle(void *pdata)
{
AD4 = AD_GetValue(ADC_Channel_4); //单次启动ADC,转换通道4
AD5 = AD_GetValue(ADC_Channel_5); //单次启动ADC,转换通道5
Voltage4 = (float)((AD4/10)*72) / 4095 * 3.3; //将AD值线性变换到0~3.3的范围,表示电压
Voltage5 = (float)((AD5/10)*72)/ 4095 * 3.3; //将AD值线性变换到0~3.3的范围,表示电压
printf("Voltage4=%.2f\r\n",Voltage4);//外部电源
printf("Voltage5=%.2f\r\n",Voltage5);//电池
vTaskDelay(200);
}
这是看B站的大神的课学会的代码:江协科技
然后结合自己的原理图改动的代码