最近希望恢复性学习一下STM8的相关知识,于是我选择了从头开始写温湿度传感器DHT11驱动代码的方式。其中遇到一些问题,也有一些收获,希望会帮助到遇到类似问题的朋友,也希望不足之处得到大家的指导。
首先介绍一下DHT11的必要知识
一 复位时序 以及 数据时序
下面是数据时序
此外,根据数据手册得知,一次通信需要的时间是3毫秒左右,这很重要,在后面的BUG分析环节会说到
二 贴上关键代码以及分析
//复位DHT11
voidDHT11_RST()
{
TIM4_CR1=0x00;//关闭定时器
TIM4_CNTR=0;//保证下次的第一个数据位的准确
DATA_SET;//ODR设置为1
DATA_OUT();//推挽输出模式,此时输出高电平
DATA_CLR;//此时处于主机输出模式,总线拉低
TIM2_Delayus(20000);//拉低20毫秒
DATA_SET;//释放总线
TIM2_Delayus(40);//释放总线以后等待40微秒DHT会发出响应信号
}
//检测DHT11是否响应
ucharDHT11_CHECK()
{
if(!DATA_GET)//如果顺利拉低,就说明有了响应
{
while((!DATA_GET)&&(outline《100))//先是低电平
{
TIM2_Delayus(1);
}
if(outline》90)//起始信号超时退出
return0;
outline=0;
while((DATA_GET)&&(outline《100))//接着是高电平
{
TIM2_Delayus(1);
}
if(outline《90)
TIM4_CR1=0x81;//立刻打开定时器开始计时第一个数据位
else
return0;
DATA_IN();//引脚设置为外部中断模式
outline=0;
return1;//一切成功返回1
}
else
return0;
}
#pragmavector=0x05//PA的中断向量位
__interruptvoidGPIOA_IRQHandler()
{
datatime=TIM4_CNTR;//获取两次下降沿之间的数据宽度
TIM4_CNTR=0;//清零,再次获取下一位
datareg《《=1;//高位先出,左移 *** 作
if((datatime》75)&&(datatime《85))//数据0我就默认高位开始获取了
datareg&=0xfe;
if((datatime》120)&&(datatime《130))//数据1
datareg|=0x01;
if(datanum==7)
dataall[0]=datareg;//获取第一个字节也就是湿度整数位
if(datanum==23)//获取第三个字节也就是温度整数位
dataall[1]=datareg;
if(datanum==39)//获取第五个字节也就是校验(温度+湿度)位
dataall[2]=datareg;
datanum++;//每次读取一位进1
if(datanum》=40)//数据接收完了结束
datanum=0;
}
三 总结以及BUG分析
总的来说 这是一款使用起来非常简单的传感器,但是作为菜鸟的我依旧是遇到了好多的问题
BUG 1 Q: 复位完毕以后,DHT11拉低总线然后再度拉高之后就不再拉低,不出数据
A: 因为在之前的程序中,我喜欢在DHT拉低以后用串口发送一个“0 FINISH”来标记DHT的引脚响应情况,而且这样也显得很叼。可是之前说过了,一次DHT的数据通信大概就3毫秒,可是你知道串口发送字符串是一件多么努力而且费时间的事情吗,你把人家DHT最好的年华都错过了啊,当你再次读取高电平的时候,对不起,这已经是数据通信结束的事情了。所以,单总线时序中不要加入一些影响读取时序的代码。
BUG 2 Q:用下降沿获取数据位数的时候,发现触发非常多,而且无论如何修改触发方式都无法改变这一现状
A:这里要说到一个之前不知道的小知识,EXTI_CR寄存器只有在总中断关闭的是时候才可以修改,所以之前一直无法修改,默认的进行了下降沿以及低电平触发的方式。当然失败了。至于其他寄存器是不是也这样就不得而知了。在之后的学习中会慢慢记住的。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)