记录一下STM32的ADC编程方法!
前面已经学习了DMA,知道如何使用DMA去减小CPU的负担,这里的ADC转换也来使用DMA---这个也是STM32的ADC转换最常见的方式。
---第一步是---了解STM32的ADC对应的GPIO口----如下图---不用记住,可以查询,我是将它剪下来粘贴到书本的相应章节----!
---第二步是---配置相应ADC转换的GPIO口----这里使用PC0--PC1
static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //打开DMA1的时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 |GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入
GPIO_Init(GPIOC, &GPIO_InitStructure);
}
---第三步是---配置ADC的DMA----配置ADC通道等---
#define ADC1_DR_Address ((u32)0x40012400+0x4c) //外设地址
__IO uint16_t ADC_ConvertedValue[2]; //内存数组
static void ADC1_Mode_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
DMA_DeInit(DMA1_Channel1);
//---------------ADC的DMA配置--------------------
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; //ADC1地址---代表ADC1保存转换值的寄存器
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue; //内存地址---用来保存DMA传输过来的ADC转换值----后面直接使用的变量地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //外设为数据源
DMA_InitStructure.DMA_BufferSize = 2; //传输总数据---2通道需要传输2个数据
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址固定
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址自增---总体表示始终从外设ADC1地址处取值---依次保存到连续的两个内存变量中---
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //外设传输数据单元---半字16位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //内存传输数据单元---半字16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //循环模式---2个数据依次循环接收从外设ADC1传输过来的ADC值---
DMA_InitStructure.DMA_Priority = DMA_Priority_High; //高优先级
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //禁止内存传内存
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
DMA_Cmd(DMA1_Channel1, ENABLE); //再次打开DMA1
//------------ADC模式配置------------------------
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //独立模式----还有很多模式---这个比较常见
ADC_InitStructure.ADC_ScanConvMode = ENABLE ; //扫描模式---采集多通道使用----本程序采集2通道---所以扫描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //连续转换模式---不难理解---就是不停地采集---一次接一次
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //不使用外部触发转换---触发分为外部触发---比如中断与定时器。软件触发---后面有专用函数
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //采集的数据右对齐---方便计算
ADC_InitStructure.ADC_NbrOfChannel = 2; //总共需要转换的通道个数---这里2个
ADC_Init(ADC1, &ADC_InitStructure);
RCC_ADCCLKConfig(RCC_PCLK2_Div8); //配置ADC转换时钟---PCLK2的8分频
//下面这个函数比较重要----配置ADC的通道与采样周期---前面说的PC0与PC1对应的ADC通道分别是--10与11。采集周期也有几种。
ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTime_55Cycles5);
ADC_DMACmd(ADC1, ENABLE); //打开DMA1的ADC1
ADC_Cmd(ADC1, ENABLE); //打开ADC1
ADC_ResetCalibration(ADC1); //复位校准寄存器
while(ADC_GetResetCalibrationStatus(ADC1)); //等待校准寄存器复位完成
ADC_StartCalibration(ADC1); //ADC校准
while(ADC_GetCalibrationStatus(ADC1)); //校准完成
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //软件触发转换
}
---第四部分是---在硬件上使用了一个通道切换芯片----CD4052----由PC2---PC3控制通道的选择CD4052切换控制GPIO配置----
void GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推完输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOC, &GPIO_InitStructure);
}
---第五部分是---主函数-----
extern __IO uint16_t ADC_ConvertedValue[2]; //声明外部变量
uint16_t My_ADC[2]; //求平均值
int main(void)
{
u8 i,led=0x01;
USART1_Config();
ADC1_GPIO_Config();
ADC1_Mode_Config();
while (1)
{
My_ADC[0]=0;
My_ADC[1]=0;
for(i=0;i<10;i++)
{
My_ADC[0]+=ADC_ConvertedValue[0];
My_ADC[1]+=ADC_ConvertedValue[1];
}
My_ADC[0]=My_ADC[0]/10; //采集10次求平均值
My_ADC[1]=My_ADC[1]/10;
ADC_ConvertedValueLocal =(float) My_ADC[0]/4096*3.3; //转换为电压值
printf("rn The current AD---0 value = 0x%04X n", My_ADC[0]);
printf("The current AD---0 value = %f V n",ADC_ConvertedValueLocal);
ADC_ConvertedValueLocal =(float) My_ADC[1]/4096*3.3;
printf("The current AD---1 value = 0x%04X n", My_ADC[1]);
printf("The current AD---1 value = %f V n",ADC_ConvertedValueLocal);
}
}
上一篇:STM32 中断配置 自我学习总结
下一篇:STM32 I2C总线 自我总结学习
推荐阅读最新更新时间:2024-11-10 11:06
推荐帖子
- 飞龙教程------51单片机
- 飞龙教程------51单片机51单片机=====单片机的学习http://v.ku6.com/show/2HAQ5tLXh4rUFyto.html51单片机=====跑马灯(LED)http://v.ku6.com/show/ki1nwsD0YCYnrQX7.html51单片机=====色环电阻识别方法http://v.ku6.com/show/9ZxwxSeT2WdxebAS.html51单片机=====蜂鸣器、http://v.ku6.com/show/xHdmQxBPAA
- smartcool 嵌入式系统
- 空载保护电路具体的工作原理
- 空载保护电路具体的工作原理,比如输入正弦波幅值50V,峰峰值100VPP,频率1.7M,则有效值检测芯片输出的有效值电压是多少?空载保护电路具体的工作原理有效值是除以根号2吗~我猜的~问题与题目没有关联吧正弦波峰值为有效值的2倍。峰峰值为2倍的峰值AD637就是一个有效值检测芯片好好看看手册就知道了至于所谓的空载保护电路具体的工作原理楼主的截的一片图,人家论文作者已经说的很清楚 就是没看懂嘛
- QWE4562009 电路观察室
- 无法找到ntifs.h 头文件,搞了两天了。都吐了高手帮忙啊
- 开发环境IFS2003+VS2008+DDKWIZARD开发普通的驱动都很好。可是有一天我想尝试一下文件驱动:我把sfilter里面的代码复制到我自己建的一个驱动项目里面,调好了编译环境,一编译找不到ntifs.h文件,于是我就把Include的路径改了一下,结果这下好了。找到了那个头文件,可是出了100多个错误,这怎么解决,高手有没有知道的,帮帮忙啊。无法找到ntifs.h头文件,搞了两天了。都吐了高手帮忙啊另外补充一下,我不用VS2008的话直接在CMD中用Build编译却是正确的
- zhtwn 嵌入式系统
- 用g2553测量外部直流电压并LCD1602显示
- 捣鼓好几天了没有一点进展啊求大神给我发个能实现用g2553测量外部电压并LCD1602显示的程序,不胜感激用g2553测量外部直流电压并LCD1602显示程序是需要依赖硬件支持的,不懂原理图,就算别人发给你了程序,你能懂吗?在编写程序的时候,都是一步一步来的,你的困难在哪?是卡在了哪个地方,并且说出你的看法,这样解决问题才是最快速的。“捣鼓好几天了……”,把平台搭建好了没有?单片机可以运行了吗?首先做个IO口控制的,能运行后,看看1602(或下载一个程序再自己修改一下),把LC
- 不绝望的面对 微控制器 MCU
- 直播已结束| Littelfuse【如何提高智能楼宇电子设备的安全与可靠性】
- 直播详情:Littelfuse有奖直播:如何提高物联网时代智能楼宇电子设备的安全与可靠性直播主题:如何提高物联网时代智能楼宇电子设备的安全与可靠性直播时间:2019年8月15日(今天)上午10:00-11:30观看方式:点击我要观看,填写并提交报名表即可。观看直播或者直播中提问均有机会获赠下方精美礼物呦~直播内容:在技术快速发展的时代给我们生活带来更多的便利和舒适,同时新技术产品也会带来一些潜在的风险和瑕疵。本次
- EEWORLD社区 安防电子
- 硕士论文
- 基于Lyapunov稳定性定理硕士论文
- lorant FPGA/CPLD
设计资源 培训 开发板 精华推荐
- ADIS16209/PCBZ,基于 ADIS16209 加速度计倾角计/加速度计的 iSensor 评估板
- 相位/串行输入2相步进电机驱动IC —— TB67S521FTAG
- MAXREFDES1204:使用MAX16990的单路输出SEPIC转换器
- LTC3624EMSE-23.3 3.3V 输出电压、2A 同步降压型稳压器的典型应用,具有 1MHz、突发模式操作
- LT3970EMS-3.3 2.5V 降压转换器的典型应用
- MC78M06BDTRKG 6V 可调输出稳压器的典型应用
- LT3091EFE GND 引脚参考 SHDN 信号的典型应用
- 具有可编程电流限制的 LT3091EFE 线性稳压器的典型应用
- 具有突发模式操作的 LTC3130EUDC-1 宽输入电压范围 15V 转换器的典型应用
- AM1G-1209DZ ±9V 1 瓦 DC/DC 转换器的典型应用