stm32f103用屏幕显示MQ135和MQ4传感器的数据有问题
MQ135能正常显示就是MQ4的数值刚开始是正常的,当我改变甲烷值的时候MQ4的数据就出问题了
main.c
#include "MQ_Sensor.h"
#include "stm32f10x.h"
#include "stm32f10x_conf.h"
#include "stm32f10x_exti.h"
#include <math.h>
#include <stddef.h>
#include <stdint.h>
#include <string.h>
#include "stm32f10x_gpio.h"
#include "stm32f10x_it.h"
#include "sys.h"
#include "delay.h"
#include "sys/types.h"
#include "usart.h"
#include "led.h"
#include "myiic.h"
#include "time.h"
#include "LCD_calculate.h"
#include "Lcd_Driver.h"
#include "dht11.h"
#include "adc.h"
#include "stdlib.h"
//光照阈值
#define lux 2200
//光敏电阻输入值
unsigned int lux_value = 0;
char temp_buff[20], temp_buff2[20];
unsigned temptempdata = 0;
int main(void)
{
char vol[25];
char vol1[25];
float ppm = 0;
float ppm1 = 0;
char txt[32];
char txt1[32];
delay_init(); //延时函数初始化
TIME_Init(); //初始化定时器
LED_Init(); //初始化RGB灯
Lcd_Init(); // LCD屏幕初始化
Lcd_Clear(GRAY0); //清屏
Redraw_Mainmenu(); //主页显示初始化
DHT11_Init(); // DHT11温湿度
Adc_Init(); //MQ135和MQ4
while (1)
{
GetMQ4Out();
GetMQ135Out();
if (timecount2 > 10) // 1000ms更新一次数据
{
timecount2 = 0;
DHT11_Read_Data(&tempdata, &humidata); //读取温湿度值
sprintf(txt, "%dC ", tempdata); //写入数据
Gui_DrawFont_GBK16(85, 27, GREEN, BLUE, txt);//显示数据
sprintf(txt1, "%d%%RH", humidata); //写入数据
Gui_DrawFont_GBK16(85, 50, GREEN, BLUE, txt1);//显示数据
ppm = GetMQ135Out();//转换为空气污染指数
sprintf(vol, "%0.1f ", ppm); //写入数据
Gui_DrawFont_GBK16(85, 73, GREEN, BLUE, vol);//显示数据
ppm1 = GetMQ4Out(); //获取MQ4数据
sprintf(vol1, "%0.1f ", ppm1); //写入数据
Gui_DrawFont_GBK16(85, 140, GREEN, BLUE, vol1);//显示数据
}
}
}
adc.c
#include "adc.h"
#include "delay.h"
#include "stm32f10x_gpio.h"
void Adc_NVIC_Config(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
NVIC_InitStructure.NVIC_IRQChannel = ADC1_2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
//初始化ADC
//这里我们仅以规则通道为例
//我们默认将开启通道1
void Adc_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1 , ENABLE ); //使能ADC1通道时钟
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M
//PA6 作为模拟通道输入引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOA, &GPIO_InitStructure);
ADC_DeInit(ADC1); //复位ADC1
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC工作模式
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数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC通道的数目
ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
ADC_Cmd(ADC1, ENABLE);//使能指定的ADC1
ADC_ResetCalibration(ADC1); //使能复位校准
while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束
ADC_StartCalibration(ADC1); //开启AD校准
while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束
//ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE);
//ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, ADC_SampleTime_1Cycles5);
//ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
//获得ADC值
//ch:通道值 1
u16 Get_Adc(u8 ch)
{
//设置指定ADC的规则组通道,一个序列,采样时间
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_1Cycles5 );//ADC1,ADC通道,采样时间为239.5周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束
return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果
}
u16 Get_Adc_Average(u8 ch,u8 times)
{
u32 temp_val=0;
u8 t;
for(t=0;t<times;t++)
{
temp_val+=Get_Adc(ch);
delay_ms(1);
}
return temp_val/times;
}
void ADC1_2_IRQHandler(void)
{
if (ADC_GetITStatus(ADC1, ADC_IT_EOC) == SET)
{
}
ADC_ClearITPendingBit(ADC1, ADC_IT_EOC);
}
// 存放 ADC 通道采样值
volatile unsigned short int ADC_ConvertedValue[2] ;
void ADC1_DMA_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
// 开启 DMA 时钟
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
// 开启 ADC 时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
// 开启 GPIO 时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
/* * * * * * ADC 引脚配置 * * * * * */
// PA0/1/2/3/4 作为模拟通道输入引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOA, &GPIO_InitStructure);
/* * * * * * DMA 模式配置 * * * * * */
// 复位 DMA 控制器
DMA_DeInit(DMA1_Channel1);
// 源数据地址
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(ADC1->DR);
// 目标地址
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADC_ConvertedValue;
// 方向:外设到存储器
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
// 传输大小
DMA_InitStructure.DMA_BufferSize = 5;
// 外设地址递增
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
// 内存地址递增
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
// 外设数据单位
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
// 内存数据单位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
// DMA模式,循环
// DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
// 优先级
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
// 关闭内存到内存的传输
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
// 初始化DMA
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
// 使能DMA
DMA_Cmd(DMA1_Channel1, ENABLE);
/* * * * * * ADC 模式配置 * * * * * */
// ADC_DeInit(ADC1); //将外设 ADC1 的全部寄存器重设为缺省值
// 只使用一个ADC,属于独立模式
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
// 扫描模式
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
// 连续转换模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
// 转换由软件开启
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
// ADC数据右对齐
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
// 顺序进行规则转换的ADC通道的数目
ADC_InitStructure.ADC_NbrOfChannel = 2;
// 根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
ADC_Init(ADC1, &ADC_InitStructure);
// 配置ADC时钟N狿CLK2的8分频,即9MHz
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
ADC_RegularChannelConfig( ADC1, ADC_Channel_0, 1, ADC_SampleTime_1Cycles5);
ADC_RegularChannelConfig( ADC1, ADC_Channel_1, 2, ADC_SampleTime_1Cycles5);
// 使能 ADC1 DMA 请求
ADC_DMACmd(ADC1, ENABLE);
// 使能 ADC1, 开始转换
ADC_Cmd(ADC1, ENABLE);
ADC_ResetCalibration(ADC1); //重置指定的ADC1的校准寄存器
while(ADC_GetResetCalibrationStatus(ADC1)); //获取ADC1重置校准寄存器的状态,设置状态则等待
ADC_StartCalibration(ADC1); //开始指定ADC1的校准状态
while(ADC_GetCalibrationStatus(ADC1)); //获取指定ADC1的校准程序,设置状态则等待
// ADC1的软件转换启动
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
MQ_Sensor.c
#include "MQ_Sensor.h"
static unsigned int adc_value = 0; //保存ADC数值
#define CAL_PPM 20
#define RL 10.0f
static float R0;
int Calibration = 1;
void MQ4_PPM_Calibration(float RS)
{
R0 = RS / pow(CAL_PPM / 613.9f, 1 / -2.074f);
}
//MQ4数据获取
float GetMQ4Out(void)
{
float RS;
double volt = 0;
double ppm = 0;
delay_ms(100);
//adc数值
adc_value = Get_Adc(1);
volt=(double)adc_value*3.3/4095;
RS = (3.3f - volt) / volt * RL;
if (Calibration)
{
MQ4_PPM_Calibration(RS);
// delay_ms(50);
Calibration = 0;
}
ppm = 613.9f * pow(RS/R0, -2.074f);
return ppm;
}
//MQ135数据获取
float GetMQ135Out(void)
{
double volt = 0;
double ppm = 0;
delay_ms(20);
//adc数值
adc_value = Get_Adc(1);
printf("adcvalue = %d.\n",adc_value);
//adc输出电压
volt=(double)adc_value*3.3/4095;
printf("volt = %f.\n",volt);
//根据电压值换算成被检测气体的污染指数
// MQ135 MQ8 H2 空气质量 通用
// volt = ((3.65*pow(ppm,0.3203*2))/(34.88+pow(ppm,0.3203*2))+0.6);
ppm = pow(10, (log10(34.88*(volt-0.6)/(3.65-volt+0.6)))/(0.3203*2));
printf("ppm = %f.\n\n",ppm);
return ppm;
}