MQ-8氢气传感器STM32驱动移植实战:ADC与GPIO双模式数据采集详解
MQ-8氢气传感器STM32驱动移植实战ADC与GPIO双模式数据采集详解最近在做一个环境监测的小项目需要检测氢气浓度就用上了MQ-8传感器。这个模块挺有意思它同时提供了模拟量AO和数字量DO两种输出意味着我们可以用STM32的ADC读取精确的浓度值也可以用GPIO直接判断是否超标非常灵活。但在实际移植驱动到立创开发板STM32F103时还是遇到了一些配置上的小问题。今天我就把整个移植过程、代码实现以及需要注意的坑点手把手地分享给大家目标是让你看完就能在自己的项目里用起来。1. 认识MQ-8传感器它如何工作在写代码之前咱们先花两分钟了解一下MQ-8传感器是怎么“感觉”到氢气的。这能帮你更好地理解后续的代码逻辑。MQ-8传感器的核心是一层特殊的材料——二氧化锡SnO₂。这种材料在干净的空气里导电能力比较差电阻大。但是当它周围存在氢气时神奇的事情发生了材料的电导率会随着氢气浓度的增加而变大电阻变小。模块内部已经集成了一个简单的电路它负责把这个电阻的变化转换成我们可以测量的电压信号。这就是AO模拟输出引脚信号的来源——电压高低直接对应着氢气浓度的高低。模块上还有一个DO数字输出引脚它又是干嘛的呢其实模块上还有一个叫LM393的电压比较器芯片。它会把AO的电压和一个可以调节的阈值电压通过板载的蓝色电位器旋钮调节进行比较。如果AO电压高于阈值意味着浓度超标DO就输出高电平反之则输出低电平。所以DO引脚给了我们一个“开关量”的报警信号用起来非常方便。简单总结一下两个引脚的区别AO引脚模拟量输出一个0-VCC之间的电压值需要接STM32的ADC引脚来读取可以得到具体的浓度比例信息。DO引脚数字量输出一个高电平或低电平需要接STM32的普通GPIO引脚来读取只能判断“有”或“无”超标。模块工作电压是3.3V-5V和咱们的3.3V单片机系统完全兼容直接连接就行。2. 硬件连接与引脚规划拿到模块上面有4个引脚VCC, GND, DO, AO。连接非常简单VCC- 开发板的3.3VGND- 开发板的GNDAO-STM32的ADC输入引脚我们例程里用的是PA5对应ADC1的通道5DO-STM32的普通GPIO输入引脚我们例程里用的是PA1这里有个关键点需要你根据自己板子的情况来定AO引脚必须连接到具有ADC功能的STM32引脚上。比如STM32F103C8T6它的PA0-PA7, PB0, PB1等引脚都支持ADC。你完全可以根据自己板子的布线方便程度来更换只要在代码里同步修改引脚定义就行。DO引脚则宽松很多任何普通的GPIO口都可以。我们的目标就是编写程序同时读取这两个引脚的数据。3. 代码移植与驱动编写移植驱动其实就是创建两个文件bsp_mq8.c函数实现和bsp_mq8.h引脚和函数声明。咱们先看头文件它定义了所有用到的硬件资源。3.1 引脚与参数定义 (bsp_mq8.h)头文件就像一份“配置清单”把硬件连接关系告诉编译器。#ifndef _BSP_MQ8_H_ #define _BSP_MQ8_H_ #include stm32f10x.h // 1. 时钟定义GPIO和ADC挂载在哪条总线上 #define RCC_MQ_GPIO RCC_APB2Periph_GPIOA // GPIOA的时钟在APB2上 #define RCC_MQ_ADC RCC_APB2Periph_ADC1 // ADC1的时钟也在APB2上 // 2. ADC外设与通道定义 #define PORT_ADC ADC1 // 使用ADC1模块 #define CHANNEL_ADC ADC_Channel_5 // 使用ADC1的第5通道 // 3. GPIO端口与引脚定义 #define PORT_MQ_GPIO GPIOA // AO和DO都接在GPIOA上 #define GPIO_MQ_AO GPIO_Pin_5 // AO接在PA5它也是ADC1_CH5 #define GPIO_MQ_DO GPIO_Pin_1 // DO接在PA1 // 4. 采样参数 #define SAMPLES 30 // ADC采样次数用于软件滤波 // 5. 函数声明 void ADC_MQ8_Init(void); unsigned int Get_Adc_MQ8_Value(void); unsigned int Get_MQ8_Percentage_value(void); char Get_MQ8_DO_value(void); #endif关键点说明如果你的AO引脚接在了PA6ADC1通道6那么你需要把CHANNEL_ADC改为ADC_Channel_6把GPIO_MQ_AO改为GPIO_Pin_6。SAMPLES定义为30意味着我们会连续采样30次然后取平均值这是一种简单的软件滤波可以让读到的数值更稳定减少跳动。3.2 初始化函数配置ADC与GPIO (bsp_mq8.c)这是最核心的部分我们来一步步拆解ADC_MQ8_Init这个初始化函数。void ADC_MQ8_Init(void) { // 第一步打开时钟开关 RCC_APB2PeriphClockCmd (RCC_MQ_GPIO, ENABLE); // 使能GPIOA的时钟 RCC_APB2PeriphClockCmd (RCC_MQ_ADC, ENABLE); // 使能ADC1的时钟 // 第二步配置AO引脚PA5为模拟输入模式 GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin GPIO_MQ_AO; GPIO_InitStructure.GPIO_Mode GPIO_Mode_AIN; // 特别注意ADC引脚必须配置为模拟输入(AIN) GPIO_Init(PORT_MQ_GPIO, GPIO_InitStructure); // 第三步配置DO引脚PA1为上拉输入模式 GPIO_InitStructure.GPIO_Pin GPIO_MQ_DO; GPIO_InitStructure.GPIO_Mode GPIO_Mode_IPU; // 上拉输入引脚悬空时默认为高电平 GPIO_Init(PORT_MQ_GPIO, GPIO_InitStructure); // 第四步复位ADC1让它恢复到默认状态 ADC_DeInit(PORT_ADC); // 第五步配置ADC的工作参数 ADC_InitTypeDef ADC_InitStruct; ADC_InitStruct.ADC_Mode ADC_Mode_Independent; // 独立模式单ADC工作 ADC_InitStruct.ADC_ScanConvMode DISABLE; // 非扫描模式因为我们只用了1个通道 ADC_InitStruct.ADC_ContinuousConvMode ENABLE; // 开启连续转换ADC会不停地采样 ADC_InitStruct.ADC_ExternalTrigConv ADC_ExternalTrigConv_None; // 软件触发不需要外部信号 ADC_InitStruct.ADC_DataAlign ADC_DataAlign_Right; // 数据右对齐方便我们读取 ADC_InitStruct.ADC_NbrOfChannel 1; // 规则通道序列里只有1个通道 ADC_Init(PORT_ADC, ADC_InitStruct); // 将配置写入ADC寄存器 // 第六步配置ADC时钟。STM32的ADC时钟不能超过14MHz这里对系统时钟72MHz进行6分频得到12MHz RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 第七步指定ADC用哪个通道以及采样时间 // 参数ADC1, 通道5, 采样顺序为1采样周期为55.5个ADC时钟周期 // 采样时间越长精度一般越高但转换速度越慢。对于传感器这种慢速信号55.5周期是常用值。 ADC_RegularChannelConfig(PORT_ADC, CHANNEL_ADC, 1, ADC_SampleTime_55Cycles5); // 第八步使能ADC ADC_Cmd(PORT_ADC, ENABLE); // 第九步ADC校准非常重要不校准读数可能不准 ADC_ResetCalibration(PORT_ADC); // 启动复位校准 while (ADC_GetResetCalibrationStatus(PORT_ADC)); // 等待复位校准完成 ADC_StartCalibration(PORT_ADC); // 启动校准 while (ADC_GetCalibrationStatus(PORT_ADC)); // 等待校准完成 // 第十步启动连续转换 ADC_SoftwareStartConvCmd(PORT_ADC, ENABLE); }注意ADC校准是必须的步骤每次上电后ADC的内部电路会有微小偏差校准就是让它自我修正。如果跳过这一步采集到的电压值可能会有几十个字的误差。3.3 数据读取函数初始化完成后我们就可以愉快地读取数据了。驱动里提供了三个函数。函数1获取原始ADC值这个函数负责从ADC数据寄存器里读取数值并且进行了软件平均滤波。unsigned int Get_Adc_MQ8_Value(void) { uint32_t Data 0; // 循环采样SAMPLES次我们在头文件里定义为30次 for(int i 0; i SAMPLES; i) { Data ADC_GetConversionValue(PORT_ADC); // 读取当前ADC转换值 delay_ms(5); // 每次读取间隔5ms避免连续读取相同值 } Data Data / SAMPLES; // 计算平均值 return Data; // 返回12位ADC原始值范围0-4095 }函数2将ADC值转换为百分比这个函数把原始的0-4095的ADC值转换成更直观的0-100%的浓度百分比。这里假设4095对应传感器最大输出电压即VCC3.3V0对应0V。unsigned int Get_MQ8_Percentage_value(void) { int adc_max 4095; // STM32的12位ADC满量程值 int adc_new 0; int Percentage_value 0; adc_new Get_Adc_MQ8_Value(); // 先获取滤波后的ADC值 // 将ADC值转换为百分比。注意这里用了强制类型转换为float保证计算精度。 Percentage_value ((float)adc_new / (float)adc_max) * 100.f; return Percentage_value; }提示这个百分比是“电压百分比”而不是“氢气浓度百分比ppm”。要得到真实的浓度你需要根据传感器数据手册的曲线图将这个电压值代入公式进行计算。这里提供的百分比输出主要用于相对浓度的观察和阈值判断。函数3读取数字输出DO状态这个函数就简单多了直接读取GPIO引脚的电平。char Get_MQ8_DO_value(void) { // 读取PA1引脚的电平 if( GPIO_ReadInputDataBit(PORT_MQ_GPIO, GPIO_MQ_DO) RESET ) // 如果读到低电平 { return 0; // 返回0表示浓度未超过模块上设定的阈值 } else // 读到高电平 { return 1; // 返回1表示浓度已超过阈值 } }模块上的蓝色电位器就是用来调节这个触发阈值的。顺时针旋转灵敏度降低需要更高浓度才触发逆时针旋转灵敏度升高。4. 在主函数中调用与验证驱动写好了最后就是在主函数里调用它们并且把数据打印出来看看。这里我们以串口打印为例。#include stm32f10x.h #include board.h // 你的板级支持包包含系统初始化、延时函数等 #include bsp_uart.h // 你的串口初始化函数 #include stdio.h // 用于printf #include bsp_mq8.h // 我们刚才写的MQ-8驱动头文件 int main(void) { board_init(); // 开发板硬件初始化系统时钟、GPIO等 uart1_init(115200U); // 初始化串口1波特率115200用于打印 ADC_MQ8_Init(); // 初始化MQ-8传感器所需的ADC和GPIO printf(MQ-8 Hydrogen Sensor Demo Start\r\n); while(1) { // 读取并打印氢气浓度百分比 printf(Hydrogen Level %d%%\r\n, Get_MQ8_Percentage_value()); // 读取并打印数字报警状态 if(Get_MQ8_DO_value() 1) { printf(Warning: Hydrogen concentration EXCEEDED threshold!\r\n); } else { printf(Status: Normal\r\n); } printf(\r\n); // 空行分隔 delay_ms(1000); // 每秒读取一次 } }将程序编译下载到开发板打开串口助手如XCOM你就能看到每秒输出一次的传感器信息。用手轻轻向传感器吹气注意不要直接哈气水汽会影响传感器或者用打火机释放少量气体务必注意安全远离明火观察ADC百分比值的变化和DO状态的变化。5. 实战中的常见问题与调试技巧ADC读数一直为0或4095检查硬件连接首先确认VCC、GND是否接对AO引脚是否确实连接到了STM32的ADC引脚如PA5。检查引脚模式确保在初始化时AO引脚配置为了GPIO_Mode_AIN模拟输入这是最容易出错的地方。检查ADC校准确认ADC_MQ8_Init函数中的校准步骤被执行了且两个等待循环都正常通过。ADC读数跳动很大软件滤波我们代码里已经做了30次取平均如果还跳动可以尝试增大SAMPLES的次数比如增加到50或100。硬件滤波在传感器的AO输出引脚和地之间并联一个0.1uF-1uF的电容可以很好地滤除高频噪声。电源噪声确保给传感器供电的3.3V电源干净稳定。可以用示波器看看电源纹波。DO输出状态不对调节电位器用螺丝刀旋转模块上的蓝色电位器改变比较器的阈值。这是调节灵敏度的主要方法。检查GPIO模式DO引脚配置为了上拉输入GPIO_Mode_IPU如果模块输出高电平STM32应读到1模块输出低电平导通到地STM32内部上拉被拉低读到0。如果反了可以尝试改为浮空输入GPIO_Mode_IN_FLOATING但更推荐上拉输入状态更稳定。如何将电压百分比转换为真实的氢气浓度ppm这是进阶应用。你需要找到MQ-8的数据手册里面会有一张“灵敏度特性曲线图”。图中横坐标是浓度ppm纵坐标是传感器电阻比Rs/R0。你需要在洁净空气中测量传感器的输出电压计算出此时的电阻R0。在实际测量中根据测得的电压计算出Rs。计算Rs/R0的比值然后去曲线图上查找对应的浓度值。这个过程需要一定的计算和拟合网上有很多关于MQ系列传感器的浓度换算代码可以参考。好了关于MQ-8传感器在STM32上的双模式驱动移植核心内容就是这些。代码我已经在实际的立创开发板上跑通了你可以直接拿去用。最关键的是理解ADC的配置流程和双模式数据采集的思路。遇到问题多检查硬件连接和软件配置一步步调试总能解决的。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2414793.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!