2571|0

23

帖子

0

TA的资源

一粒金砂(中级)

楼主
 

【 STM32WB55 测评】_06_温湿度数据上传实验 [复制链接]

本次活动测评开发板STM32WB55 Nucleo Pack由ST意法半导体提供,感谢意法半导体对EEWorld测评的支持!
【实验目的】
· 熟悉STM32WB55的串口模块的配置与使用。
· 掌握DHT11的通信时序
· 掌握使用GPIO处理通信时序

【实验环境】
· NUCLEO-WB55 Nucleo Pack开发板
· Keil MDK-ARM(Keil uVision 5.25.2.0)
· Keil.STM32WBxx_DFP.1.0.0.pack
· 串口调试助手
· DHT11温湿度模块

【实验资料】
· NUCLEO-WB55 Nucleo Pack开发板原理图
· STM32WB55xx Data Sheets
· STM32WB55xx Reference manual(参考手册)
· DHT11用户手册
【实验分析
原理图:
连接如图所示,灰色线连接到电源的负极,黑色线连接到3.3V正极,白色是数据线,连接到PC10引脚。
DHT11 器件采用简化的单总线通信。单总线即只有一根数据线,系统中的数据交换、控制均由单总线完成。DATA 用于微处理器与 DHT11 之间的通讯和同步,采用单总线数据格式,一次传送 40 位数据,高位先出。
数据格式:8bit 湿度整数数据 + 8bit 湿度小数数据+8bit 温度整数数据 + 8bit 温度小数数据+8bit 校验位。
校验位数据定义:“8bit 湿度整数数据 + 8bit 湿度小数数据+8bit 温度整数数据 + 8bit 温度小数数据”8bit 校验位等于所得结果的末 8 位。
数据时序图
用户主机(MCU)发送一次开始信号后,DHT11 从低功耗模式转换到高速模式,待主机开始信号结束后,DHT11 发送响应信号,送出 40bit 的数据,幵触发一次信采集。
信号发送如图所示
外设读取步骤
主机和从机之间的通信可通过如下几个步骤完成(外设(如微处理器)读取 DHT11 的数据的步骤)。
步骤一:
DHT11 上电后(DHT11 上电后要等待 1S 以越过不稳定状态在此期间不能发送任何指令),测试环境温湿度数据,幵记录数据,同时 DHT11 DATA 数据线由上拉电阻拉高一直保持高电平;此时 DHT11 DATA 引脚处于输入状态,时刻检测外部信号。
步骤二:
微处理器的 I/O 设置为输出同时输出低电平,且低电平保持时间不能小于 18ms,然后微处理器的 I/O设置为输入状态,由于上拉电阻,微处理器的 I/O DHT11 DATA 数据线也随之变高,等待 DHT11 作出回答信号,发送信号如图所示:
步骤三:
DHT11 DATA 引脚检测到外部信号有低电平时,等待外部信号低电平结束,延迟后 DHT11 DATA引脚处于输出状态,输出 80 微秒的低电平作为应答信号,紧接着输出 80 微秒的高电平通知外设准备接收数据,微处理器的 I/O 此时处于输入状态,检测到 I/O 有低电平(DHT11 回应信号)后,等待 80 微秒的高电平后的数据接收,发送信号如图所示:
步骤四:
DHT11 DATA 引脚输出 40 位数据,微处理器根据 I/O 电平的变化接收 40 位数据,位数据“0”的格式为: 50 微秒的低电平和 26-28 微秒的高电平,位数据“1”的格式为: 50 微秒的低电平加 70微秒的高电平。位数据“0”“1”格式信号如图所示:
结束信号:
DHT11 DATA 引脚输出 40 位数据后,继续输出低电平 50 微秒后转为输入状态,由于上拉电阻随之变为高电平。但 DHT11 内部重测环境温湿度数据,幵记录数据,等待外部信号的到来。
【实验代码】
本实验在上一实验基础上进行添加

  1. #include <stdio.h>
  2. #include <stdint.h>
  3. #include "uart.h"
  4. void GPIOSetDir(GPIO_TypeDef *GPIO, int pin, int dir)
  5. {
  6.         if (dir == 0){
  7.                 GPIO->MODER        &= ~(0x3UL<<(pin * 2));
  8.                 GPIO->OTYPER &= ~(1<<pin);
  9.         }else{
  10.                 GPIO->MODER        &= ~(0x3UL<<(pin * 2));
  11.                 GPIO->MODER        |= (0x1UL<<(pin * 2));
  12.                 GPIO->OSPEEDR |= 0x3UL<<(pin * 2);
  13.                 GPIO->OTYPER &= ~(1<<pin);
  14.         }
  15. }

  16. void GPIOSetValue(GPIO_TypeDef *GPIO, int pin, int value)
  17. {
  18.         if (value == 0){
  19.                 GPIO->ODR &= ~(1<<pin);
  20.         }else{
  21.                 GPIO->ODR |= (1<<pin);
  22.         }
  23. }

  24. void GPIOSetToggle(GPIO_TypeDef *GPIO, int pin)
  25. {
  26.         GPIO->ODR ^= (1<<pin);
  27. }

  28. int GPIOGetValue(GPIO_TypeDef *GPIO, int pin)
  29. {
  30.         return GPIO->IDR & (1<<pin);
  31. }

  32. int DHT11_Read(char *temp, char *humd)
  33. {
  34.         int i;
  35.         int timeout = 0;
  36.         volatile int low_count, high_count;
  37.         char data[5];        //接收40位数据
  38.        
  39.         //步骤二
  40.         GPIOSetDir(GPIOC, 10, 1);
  41.         GPIOSetValue(GPIOC, 10, 0);        //输出低电平
  42.         mdelay(30);
  43.         GPIOSetDir(GPIOC, 10, 0);        //设置成输入模式
  44.         //步骤三
  45.         while(GPIOGetValue(GPIOC, 10)){ //等待响应信号
  46.                 timeout ++;
  47.                 if (timeout > 10000) return -1;
  48.         }
  49.         while(GPIOGetValue(GPIOC, 10) == 0);        //等待高电平
  50.         while(GPIOGetValue(GPIOC, 10) != 0);        //等待高电平结束
  51.         //步骤四
  52.         for (i = 0; i < 40; i ++){
  53.                 low_count = 0;
  54.                 high_count = 0;
  55.                 while(GPIOGetValue(GPIOC, 10) == 0){        //记录低电平
  56.                         low_count ++;
  57.                 }
  58.                 while(GPIOGetValue(GPIOC, 10) != 0){        //记录高电平
  59.                         high_count ++;
  60.                 }
  61.                 if (low_count > high_count){ //数据0
  62.                         data[i/8] &= ~(1<<(7-i%8));
  63.                 }else{//数据1
  64.                         data[i/8] |= (1<<(7-i%8));
  65.                 }
  66.         }
  67.         //步骤五
  68.         while(GPIOGetValue(GPIOC, 10) == 0);        //等待低电平结束
  69.        
  70.         if((data[0]+data[1]+data[2]+data[3]) == data[4]){//校验数据
  71.                 humd[0]=data[0];
  72.                 humd[1]=data[1];
  73.                 temp[0]=data[2];
  74.                 temp[1]=data[3];
  75.                 return 1;
  76.         }else{        //数据校验失败
  77.                 return 0;
  78.         }
  79. }

  80. int main(void)
  81. {
  82.         int ret;
  83.         char temp[2]={0};        //保存温度数据
  84.         char humd[2]={0};        //保存湿度数据
  85.         USART1_Init(115200);
  86.         while(1){
  87.                 ret = DHT11_Read(temp, humd);
  88.                 switch (ret){
  89.                         case -1:
  90.                                 printf("Timeout\n");
  91.                         break;
  92.                         case 0:
  93.                                 printf("Data Error\n");
  94.                         break;
  95.                         case 1:
  96.                                 printf("humd:%d.%d temp:%d.%d\n", humd[0], humd[1]
  97.                                                         , temp[0], temp[1]);
  98.                         break;
  99.                 }
  100.                 mdelay(1000);
  101.         }
  102. }
复制代码


【实验现象】
· 连接开发板,并打开串口调试助手
· 在终端中显示如下:



此内容由EEWORLD论坛网友lvxinn2006原创,如需转载或用于商业用途需征得作者同意并注明出处




此帖出自stm32/stm8论坛
点赞 关注
 

回复
举报
您需要登录后才可以回帖 登录 | 注册

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/10 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表