MQ-2烟雾传感器与立创CW32F030C8T6开发板ADC驱动移植实战
MQ-2烟雾传感器与立创CW32F030C8T6开发板ADC驱动移植实战最近在做一个智能家居的小项目需要检测厨房的烟雾浓度防止意外发生。手头正好有立创的CW32F030C8T6开发板和一个常见的MQ-2烟雾传感器模块。本以为把线一连代码一写就完事了结果在读取模拟量数据时还是遇到了一些小波折。今天我就把整个从硬件连接到软件驱动再到数据处理的完整过程梳理出来分享给同样在使用CW32或者其他Cortex-M0/M0内核MCU的朋友们。整个过程我会尽量讲得细一些特别是ADC配置和数据转换这些容易踩坑的地方。1. 认识你的“侦察兵”MQ-2烟雾传感器在写代码之前咱们得先搞清楚要驱动的对象是个啥。MQ-2传感器你可以把它理解成一个对烟雾和多种可燃气体比如液化气、天然气特别敏感的“电子鼻子”。它的核心是一层二氧化锡半导体材料。简单来说当它工作在一定的温度下传感器内部有个加热丝通电后就会加热这层材料会吸附空气中的氧气。一旦有烟雾或可燃气体过来就会和吸附的氧气发生反应导致材料的导电能力发生变化——烟雾浓度越高导电性越好电阻就越低。这个电阻变化就是我们检测烟雾浓度的关键。MQ-2模块很贴心地为我们提供了两种方式来“读取”这个变化模拟量输出AO引脚直接输出一个随烟雾浓度变化的电压值。浓度越高电压越高。这是我们获取精确浓度信息的主要通道。数字量输出DO引脚模块上有个比较器芯片通常是LM393和一个可调电阻。你可以通过旋钮设定一个报警阈值。当AO引脚电压超过这个阈值时DO引脚就会从低电平跳变成高电平给你一个“超标了”的开关信号。这个适合做简单的报警比如直接驱动一个蜂鸣器。几个关键点需要注意需要预热MQ-2内部的敏感材料需要稳定在工作温度才能准确检测。所以上电后最好等待几十秒到一分钟等它热稳定了再读取数据否则数据会漂得厉害。工作电压模块需要5V供电。注意这里说的是模块的供电电压。它的AO引脚输出的模拟电压范围一般是0-Vcc即0-5V这个电压需要接入我们MCU的ADC引脚进行测量。检测范围大概能检测100到10000ppm浓度的气体ppm是浓度单位你可以理解为每立方米空气中含有多少立方厘米的目标气体。2. 硬件连接把“鼻子”接到“大脑”上硬件连接很简单一共四根线。我用的立创CW32F030C8T6开发板连接方式如下MQ-2模块引脚连接到CW32F030开发板作用说明VCC5V引脚模块电源正极必须接5VGNDGND引脚模块电源地共地很重要AOPA5引脚模拟信号输出接MCU的ADC输入通道DOPA1引脚或其他GPIO数字开关信号输出接普通GPIO输入即可为什么选PA5因为根据CW32F030的用户手册原文提到在439页PA5引脚复用了ADC的通道5输入功能。我们就是要用MCU内部的ADC模数转换器来测量PA5引脚上的电压从而知道烟雾浓度。DO引脚我暂时接在了PA1配置成上拉输入模式。当浓度超标时读取这个引脚会是高电平。这个功能实现起来很简单本文重点我们放在更精确的AO模拟量采集上。3. 软件驱动手把手配置ADC读取数据硬件接好了接下来就是让MCU的ADC动起来。CW32的库函数封装得挺清晰我们一步步来。3.1 建立工程文件与头文件定义首先在工程里新建两个文件bsp_mq2.c和bsp_mq2.h。“bsp”意思是板级支持包专门放这块板子外设的驱动代码。我们先来看头文件bsp_mq2.h这里主要做一些宏定义把引脚、通道这些信息固定下来以后修改起来方便。#ifndef _BSP_MQ2_H_ #define _BSP_MQ2_H_ #include board.h // 包含CW32的板级基础头文件 /* 时钟使能宏定义 */ #define RCC_MQ2_GPIO_ENABLE() __RCC_GPIOA_CLK_ENABLE() // 使能GPIOA的时钟 #define RCC_MQ2_ADC_ENABLE() __RCC_ADC_CLK_ENABLE() // 使能ADC的时钟 /* 引脚模拟功能使能 */ #define GPIO_ANALOG_ENABLE() PA05_ANALOG_ENABLE() // 将PA5设置为模拟输入模式 /* ADC通道定义 */ #define MQ2_ADC_CHANNEL ADC_ExInputCH5 // 使用ADC外部通道5对应PA5 /* 端口与引脚定义 */ #define PORT_MQ2 CW_GPIOA // 使用GPIOA端口 #define GPIO_MQ2_AO GPIO_PIN_5 // AO引脚接在PA5 #define GPIO_MQ2_DO GPIO_PIN_1 // DO引脚接在PA1示例 /* 软件滤波采样次数 */ #define SAMPLES 30 // 采集30次然后求平均 /* 函数声明 */ void Adc_Init(void); // ADC初始化函数 unsigned int Get_Adc_Value(void); // 获取单次ADC转换值已滤波 unsigned int Get_MQ2_Percentage_value(void); // 获取烟雾浓度百分比值 #endif3.2 ADC初始化配置详解这是最核心的部分在bsp_mq2.c中实现。咱们把ADC初始化的代码拆开揉碎了讲。#include bsp_mq2.h void Adc_Init(void) { GPIO_InitTypeDef GPIO_InitStruct; // 定义一个GPIO初始化结构体 /* 第一步打开时钟开关 */ RCC_MQ2_GPIO_ENABLE(); // 打开GPIOA的时钟PA5属于GPIOA RCC_MQ2_ADC_ENABLE(); // 打开ADC外设的时钟不给时钟ADC是“瘫痪”的 /* 第二步配置GPIO引脚模式 */ // 虽然AO是模拟输入但DO是数字输入这里一起初始化了 GPIO_InitStruct.Pins GPIO_MQ2_AO | GPIO_MQ2_DO; // 要初始化的引脚PA5和PA1 GPIO_InitStruct.Mode GPIO_MODE_INPUT_PULLUP; // 模式上拉输入对DO有用AO模式会被后续覆盖 GPIO_InitStruct.Speed GPIO_SPEED_HIGH; // 输出速度输入模式下此配置影响不大 GPIO_Init(PORT_MQ2, GPIO_InitStruct); // 调用初始化函数 /* 第三步将PA5设置为模拟输入模式关键*/ GPIO_ANALOG_ENABLE(); // 这个宏展开就是设置PA5为模拟功能。如果不设置ADC读不到正确电压。 /* 第四步配置ADC外设参数 */ ADC_InitTypeDef ADC_InitStructure; // ADC主初始化结构体 ADC_WdtTypeDef ADC_WdtStructure; // ADC看门狗结构体本例未使用但需初始化 ADC_SingleChTypeDef ADC_SingleChStructure; // 单通道转换结构体 // 填充ADC主初始化参数 ADC_InitStructure.ADC_OpMode ADC_SingleChOneMode; // 操作模式单通道单次转换 ADC_InitStructure.ADC_ClkDiv ADC_Clk_Div4; // ADC时钟分频系统时钟64MHz / 4 16MHz ADC_InitStructure.ADC_SampleTime ADC_SampTime5Clk; // 采样时间5个ADC时钟周期 ADC_InitStructure.ADC_VrefSel ADC_Vref_VDDA; // 参考电压源选择VDDA通常接3.3V ADC_InitStructure.ADC_InBufEn ADC_BufDisable; // 输入缓冲器关闭一般情况关闭即可 ADC_InitStructure.ADC_TsEn ADC_TsDisable; // 温度传感器关闭我们测外部电压 ADC_InitStructure.ADC_DMAEn ADC_DmaDisable; // DMA关闭单次读取不用DMA ADC_InitStructure.ADC_Align ADC_AlignRight; // 数据对齐方式右对齐最常用 ADC_InitStructure.ADC_AccEn ADC_AccDisable; // 累加功能关闭不需要多次累加 // 初始化看门狗本例中未使用其功能但需按库函数要求初始化 ADC_WdtInit(ADC_WdtStructure); // 配置单通道转换的具体通道 ADC_SingleChStructure.ADC_DiscardEn ADC_DiscardNull; // 溢出保存设置 ADC_SingleChStructure.ADC_Chmux MQ2_ADC_CHANNEL; // 选择通道我们定义的通道5PA5 ADC_SingleChStructure.ADC_InitStruct ADC_InitStructure; // 传入主配置 ADC_SingleChStructure.ADC_WdtStruct ADC_WdtStructure; // 传入看门狗配置 // 调用单通道模式配置函数将以上参数写入ADC寄存器 ADC_SingleChOneModeCfg(ADC_SingleChStructure); /* 第五步使能ADC并启动转换 */ ADC_Enable(); // 使能ADC模块 ADC_SoftwareStartConvCmd(ENABLE); // 软件触发启动第一次转换 }注意ADC_Vref_VDDA意味着ADC的参考电压是VDDA引脚上的电压通常和MCU的供电电压3.3V相连。这意味着ADC测量的电压范围是0~3.3V对应的数字量输出是0~4095因为CW32F030的ADC是12位的2^124096。务必确保MQ-2模块的AO输出电压不超过3.3V否则可能损坏MCU引脚。常见的MQ-2模块输出在5V系统下是0-5V可能需要分压电阻才能接入3.3V的ADC。3.3 编写数据读取与处理函数初始化完成后我们就能读取ADC值了。但直接读一次往往噪声很大所以通常要软件滤波。/* 函数名称ADC_GET * 功能执行一次ADC转换并返回结果 * 注意这是一个底层函数被后面的滤波函数调用 */ uint32_t ADC_GET(void) { ADC_SoftwareStartConvCmd(ENABLE); // 软件触发启动一次新的转换 // 通常这里需要等待转换完成标志位但CW32库的ADC_GetConversionValue函数可能已包含等待。 uint32_t adcValue ADC_GetConversionValue(); // 从数据寄存器中读取转换结果 return adcValue; // 返回原始ADC值范围0-4095 } /* 函数名称Get_Adc_Value * 功能采集多次ADC值并求平均实现简单的软件滤波 */ unsigned int Get_Adc_Value(void) { unsigned char i 0; unsigned int AdcValue 0; // 循环采样 SAMPLES 次我们在头文件里定义为30次 for(i0; i SAMPLES; i) { AdcValue ADC_GET(); // 累加每次采样的值 } // 计算平均值 AdcValue AdcValue / SAMPLES; return AdcValue; // 返回滤波后的ADC值 } /* 函数名称Get_MQ2_Percentage_value * 功能将ADC值转换为百分比浓度值 * 原理假设ADC值线性对应电压电压线性对应浓度实际传感器非线性此处为简易演示 */ unsigned int Get_MQ2_Percentage_value(void) { int adc_max 4095; // 12位ADC最大值 int adc_new 0; int Percentage_value 0; adc_new Get_Adc_Value(); // 获取滤波后的ADC值 // 将ADC值转换为百分比(当前值 / 最大值) * 100 // 注意这里用了浮点数运算在资源紧张的MCU中可考虑用整数运算优化 Percentage_value ((float)adc_new / (float)adc_max) * 100.0f; return Percentage_value; // 返回0-100之间的百分比值 }提示这里的百分比转换是一个非常粗略的演示。在实际应用中MQ-2传感器的电阻-浓度关系是非线性的需要根据传感器数据手册中的曲线图进行校准或者使用查表法、公式法来得到更准确的ppm浓度值。Get_Adc_Value函数返回的稳定ADC值才是你进行后续校准和算法处理的“原材料”。4. 实战验证在main函数中调用并打印结果最后我们在主函数里把上面的驱动用起来并通过串口打印出结果看看传感器到底有没有在工作。#include board.h #include stdio.h // 为了使用printf #include bsp_uart.h // 串口初始化头文件需要先实现串口驱动 #include bsp_mq2.h // 我们刚才写的MQ2驱动头文件 int32_t main(void) { board_init(); // 开发板基础初始化系统时钟等 uart1_init(115200); // 初始化串口1波特率115200用于连接电脑查看数据 Adc_Init(); // 初始化ADC配置PA5引脚 printf(MQ-2 Sensor ADC Demo Start\r\n); // 主循环每隔1秒读取并打印一次数据 while(1) { // 打印原始ADC值滤波后 printf(ADC Value %d\r\n, Get_Adc_Value() ); // 打印换算后的浓度百分比 printf(MQ-2 Percentage %d%%\r\n, Get_MQ2_Percentage_value() ); printf(------------------\r\n); delay_ms(1000); // 延时1秒 } }上电后的现象将开发板通过串口连接到电脑打开串口助手工具如Xshell、SecureCRT或立创提供的工具设置好波特率115200。你就能看到每秒输出一次的数据。在清洁空气中ADC值会稳定在一个较低的基数上。当你用打火机不点燃释放一些丁烷气体或者点燃一根吹灭的香烟靠近传感器时可以看到ADC值和百分比数值明显上升。移植成功的关键点回顾引脚模式别弄错用于ADC的引脚PA5一定要设置为模拟输入模式这是最容易忽略的一步。参考电压要清楚理解ADC的测量范围0-3.3V和你的信号电压范围是否匹配不匹配需要硬件分压。采样滤波有必要ADC采样存在噪声多次采样求平均是最简单有效的软件滤波方法。传感器需要预热上电后给MQ-2一点时间30秒到1分钟等读数稳定了再作为有效数据。整个流程走下来你会发现驱动一个模拟传感器并不复杂核心就是配置好ADC然后周期性地去读取它。希望这篇实战笔记能帮你快速上手CW32F030的ADC功能并把MQ-2传感器用起来。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2409382.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!