STM32 ADC学习

 

12位ADC是一种逐次逼近型模拟数字数字转换器。它有多达18个通道,可测量16个外部和2个内部信号源。

ADC的输入时钟不得超过14MHZ,它是由PCLK2经分频产生。

如果被ADC转换的模拟电压低于低阀值或高于高阀值,AWD模拟看门狗状态位被设置。

ADC通常要与DMA一起使用 这里只是简单的用库配置ADC 不断扫描来实现ADC的应用。

首先配置GPIO与ADC的时钟:

ADC_InitTypeDef  ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

GPIO_InitStructure.GPIO_Pin  =GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_AIN;
GPIO_Init(GPIOB,&GPIO_InitStructure); //默认速度为两兆

配置ADC的运行:

ADC_InitStructure.ADC_Mode              = ADC_Mode_Independent;  //独立模式
ADC_InitStructure.ADC_ScanConvMode      =DISABLE;      //连续多通道模式
ADC_InitStructure.ADC_ContinuousConvMode =ENABLE;      //连续转换
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_RegularChannelConfig(ADC1,ADC_Channel_9, 1,ADC_SampleTime_1Cycles5); //通道X,采样时间为1.5周期,1代表规则通道第1个这个1是啥意思我不太清楚只有是1的时候我的ADC才正常。
ADC_Cmd  (ADC1,ENABLE);             //使能或者失能指定的ADC
ADC_SoftwareStartConvCmd(ADC1,ENABLE);//使能或者失能指定的ADC的软件转换启动功能

1
这里我用的是ADC1的9通道 PB1引脚。

也用一些默认的配置函数同GPIO 的一样例如: ADC_StructInit

ADC_InitTypeDef structureADC_InitTypeDef定义于文件“stm32f10x_adc.h”:

typedef struct

{

u32 ADC_Mode; FunctionalState ADC_ScanConvMode; FunctionalStateADC_ContinuousConvMode; u32 ADC_ExternalTrigConv; u32ADC_DataAlign; u8 ADC_NbrOfChannel;

} ADC_InitTypeDef

注意:为了能够正确地配置每一个ADC通道,用户在调用ADC_Init()之后,必须调用ADC_ChannelConfig()来配置每个所使用通道的转换次序和采样时间。

然后就是不停的读;

u16 TestAdc(void)
{
u16 adc;
while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC)==RESET); //检查制定ADC标志位置1与否 ADC_FLAG_EOC 转换结束标志位
adc=ADC_GetConversionValue(ADC1);

returnadc;//返回最近一次ADCx规则组的转换结果
}

这个程序的8位单片机风格很重,真正的ADC一定要放在DMA或是中断之中。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
经导师精心指导并认可、获 98 分的毕业设计项目!【项目资源】:微信小程序。【项目说明】:聚焦计算机相关专业毕设及实战操练,可作课程设计与期末大作业,含全部源码,能直用于毕设,经严格调试,运行有保障!【项目服务】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。 经导师精心指导并认可、获 98 分的毕业设计项目!【项目资源】:微信小程序。【项目说明】:聚焦计算机相关专业毕设及实战操练,可作课程设计与期末大作业,含全部源码,能直用于毕设,经严格调试,运行有保障!【项目服务】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。 经导师精心指导并认可、获 98 分的毕业设计项目!【项目资源】:微信小程序。【项目说明】:聚焦计算机相关专业毕设及实战操练,可作课程设计与期末大作业,含全部源码,能直用于毕设,经严格调试,运行有保障!【项目服务】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。 经导师精心指导并认可、获 98 分的毕业设计项目!【项目资源】:微信小程序。【项目说明】:聚焦计算机相关专业毕设及实战操练,可作课程设计与期末大作业,含全部源码,能直用于毕设,经严格调试,运行有保障!【项目服务】:有任何使用上的问题,欢迎随时与博主沟通,博主会及时解答。
以下是基于STM32F10x_StdPeriph_Lib_V3.5.0的标准库程序示例,用于将ADCPB1通道采样值转换为电压值。 ```c #include "stm32f10x.h" #define ADC1_DR_Address ((uint32_t)0x4001244C) // ADC1数据寄存器地址 void ADC1_Init(void); uint16_t Get_Adc(uint8_t ch); uint16_t Get_Adc_Average(uint8_t ch, uint8_t times); int main(void) { ADC1_Init(); // 初始化ADC1 uint16_t value = 0; while (1) { value = Get_Adc_Average(9, 10); // 获取PB1通道采样值的平均值 float voltage = (float)value * 3.3 / 4096; // 将采样值转换为电压值 // 进行电压值的处理或输出 } } /* * 函数名:ADC1_Init * 描述 :初始化ADC1 * 输入 :无 * 输出 :无 */ void ADC1_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; ADC_InitTypeDef ADC_InitStructure; DMA_InitTypeDef DMA_InitStructure; // 使能ADC1和DMA1时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); // 配置PB1为模拟输入 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOB, &GPIO_InitStructure); // DMA1通道1配置 DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&ADC_ConvertedValue; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize = 1; DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable; DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; DMA_InitStructure.DMA_Priority = DMA_Priority_High; DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(DMA1_Channel1, &DMA_InitStructure); // DMA1通道1使能 DMA_Cmd(DMA1_Channel1, ENABLE); // ADC1配置 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; ADC_InitStructure.ADC_ScanConvMode = DISABLE; ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel = 1; ADC_Init(ADC1, &ADC_InitStructure); // ADC1规则序列配置 ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 1, ADC_SampleTime_55Cycles5); // ADC1 DMA使能 ADC_DMACmd(ADC1, ENABLE); // ADC1使能 ADC_Cmd(ADC1, ENABLE); // ADC1复位校准 ADC_ResetCalibration(ADC1); while (ADC_GetResetCalibrationStatus(ADC1)) ; // ADC1开始校准 ADC_StartCalibration(ADC1); while (ADC_GetCalibrationStatus(ADC1)) ; // ADC1开始转换 ADC_SoftwareStartConvCmd(ADC1, ENABLE); } /* * 函数名:Get_Adc * 描述 :获取ADC值 * 输入 :ch通道编号 * 输出 :ADC值 */ uint16_t Get_Adc(uint8_t ch) { // ADC1规则序列设置 ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_55Cycles5); // ADC1软件启动转换 ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 等待转换结束 while (!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC)) ; // 返回转换结果 return ADC_GetConversionValue(ADC1); } /* * 函数名:Get_Adc_Average * 描述 :获取ADC采样值的平均值 * 输入 :ch通道编号,times采样次数 * 输出 :ADC采样值的平均值 */ uint16_t Get_Adc_Average(uint8_t ch, uint8_t times) { uint32_t sum = 0; uint8_t i; for (i = 0; i < times; i++) { sum += Get_Adc(ch); } return sum / times; } ``` 需要注意的是,本示例中使用了DMA方式进行ADC转换,因此需要在ADC初始化时配置DMA通道并开启DMA,同时也需要在主函数中等待DMA传输完成。同时,为了保证ADC的准确性,还需要进行ADC的校准操作。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值