dht11 + Sc32440驱动

2024-08-29 00:28
文章标签 驱动 dht11 sc32440

本文主要是介绍dht11 + Sc32440驱动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、DHT11概述

1、简介

   DHT11数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器,它应用专用的数字模块采集技术和温湿度传感技术,确保产品具有极高的可靠性和卓越的长期稳定性。传感器包括一个电阻式感湿元件和一个NTC测温元件,并与一个高性能8位单片机相连接。因此该产品具有品质卓越、超快响应、抗干扰能力强、性价比极高等优点。每个DHT11传感器都在极为精确的湿度校验室中进行校准。校准系数以程序的形式存在OTP内存中,传感器内部在检测信号的处理过程中要调用这些校准系数。单线制串行接口,使系统集成变得简易快捷。超小的体积、极低的功耗,使其成为该类应用中,在苛刻应用场合的最佳选择。产品为4针单排引脚封装,连接方便。

2、   电气特性 

3、接线

VCC接5V  
GND接GND
DATA接P3^3

4、数据传输逻辑

只有一个数据线DATA ,单片机发送序列指令给DHT11模块
模块一次完整的数据传输为40bit,高位先出。

5、数据格式:

     8bit湿度整数数据   +   8bit 湿度小数数据 
+   8bit温度整数数据   +    8bit温度小数数据 
+   8bit  校验和  

  二、DHT11时序图分析

1、通讯过程时序图

2、检测DHT11温湿度传感器是否存在
根据下面的时序图,并检测模块是否存在,功能是否正常

主机发送开始信号->主机读取响应信号->开始接收数据->主机发送停止信号。

主机发送起始信号

  • 主机(MCU)首先将总线(数据线)拉低,并保持至少18ms的低电平时间。这个低电平信号是告诉DHT11传感器,主机准备开始通信。
  • 随后,主机将总线拉高,并保持20~40μs的高电平时间。这个高电平信号的作用是告诉DHT11传感器,主机已经准备好接收数据。

DHT11响应信号

  • DHT11传感器在检测到起始信号后,会将总线拉低80μs左右,作为响应信号的开始。
  • 然后,DHT11将总线拉高80μs左右,表示响应信号的结束,并准备开始发送数据。

3、数据发送

  • DHT11传感器发送的数据总长度为40位(5字节),包括湿度整数部分(8位)、湿度小数部分(8位)、温度整数部分(8位)、温度小数部分(8位)和校验和(8位)。数据以高位先出的方式发送。
  • 每一位数据的发送都遵循以下规则:
    • 首先,DHT11将总线拉低50μs左右,表示数据位的开始。
    • 然后,根据要发送的数据位是0还是1,DHT11将总线拉高的时间会有所不同:
      • 如果发送的是0,则总线保持高电平的时间为26~28μs。
      • 如果发送的是1,则总线保持高电平的时间为70μs。

三、根据时序图写出驱动S3c2440控制DHT11

构建了一个基本的Linux内核模块,用于通过GPIO接口与DHT11温湿度传感器进行通信。这个模块通过定义一系列的函数来实现对DHT11的初始化、发送启动信号、等待响应、读取数据位以及处理整个数据包的接收

#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/module.h>
#include <asm/io.h>
#include <asm/string.h>
#include <asm/uaccess.h>
#include <linux/miscdevice.h>
#include <asm-generic/errno-base.h>
#include <mach/gpio-nrs.h>
#include <mach/gpio.h>
#include <linux/delay.h>#define DEV_NAME "dht11"
#define PIN_DHT11 S3C2410_GPF(6)static void pin_init(void)
{gpio_request(PIN_DHT11, "dht11");
}static void dht11_start(void)
{gpio_direction_output(PIN_DHT11, 0);mdelay(20);gpio_set_value(PIN_DHT11, 1);udelay(30);gpio_direction_input(PIN_DHT11);
}static int dht11_wait_repon(void)
{int time = 100;while(gpio_get_value(PIN_DHT11))   // wait H end{udelay(1);if(!time--)return -1;}time = 100;while(!gpio_get_value(PIN_DHT11) && time)  // wait 80us L end{udelay(1);if(!time--)return -2;}time = 100;while(gpio_get_value(PIN_DHT11))   // wait 80us H end{udelay(1);if(!time--)return -3;}return 0;
}static int dht11_get_bit(void)
{int time = 100;while(!gpio_get_value(PIN_DHT11))  // wait 50us L end{udelay(1);	if(!time--)return -1;}udelay(30);if(0 == gpio_get_value(PIN_DHT11))return 0;while(gpio_get_value(PIN_DHT11))   // wait 40us H end{udelay(1);	if(!time--)return -2;}return 1;
}static int dht11_get_value(unsigned char * data)
{int i = 0;int j = 0;unsigned char sum = 0;for(i = 0; i < 5; i++){data[i] = 0;for(j = 0; j < 8; j++){char bit = dht11_get_bit();	if(bit < 0)return bit;data[i] <<= 1;data[i] |= bit;}}for(i = 0; i < 4; i++){sum += data[i];}if(sum == data[4])return 4;elsereturn -1;
}static int open (struct inode * inode, struct file * file)
{pin_init();printk("dht11 open ...\n");return 0;
}static ssize_t read (struct file * file, char __user * buf, size_t len, loff_t * offset)
{unsigned char data[4] = {0};int ret = 0;dht11_start();ret = dht11_wait_repon();if(ret < 0)goto err_wait;ret = dht11_get_value(data);if(ret < 0)goto err;copy_to_user(buf, &data, sizeof(data));printk("dht11 read ...\n");return sizeof(data);err_wait:printk("dht11_wait_repon err ... \n");return ret;err:printk("dht11_get_value err ...\n");return ret;
}static ssize_t write (struct file * file, const char __user * buf, size_t len, loff_t * offset)
{return 0;
}static int close (struct inode * inode, struct file * file)
{printk("dht11 close ...\n");return 0;
}static struct file_operations fops = 
{.owner = THIS_MODULE,.open = open,.read = read,.write = write,.release = close
};static struct miscdevice misc = 
{.minor = MISC_DYNAMIC_MINOR,.name = DEV_NAME,.fops = &fops
};static int __init dht11_init(void)
{int ret = misc_register(&misc);if(ret < 0)goto err_misc_register;printk("dht11_init  ...\n");return ret;err_misc_register:misc_deregister(&misc);printk("dht11 misc_register faidht11\n");	return ret;
}static void __exit dht11_exit(void)
{misc_deregister(&misc);printk("dht11_exit  ###############################\n");
}module_init(dht11_init);
module_exit(dht11_exit);
MODULE_LICENSE("GPL");
  • dht11_start():发送启动信号给DHT11。首先拉低引脚20ms,然后拉高30μs,最后设置引脚为输入模式,等待DHT11的响应。
  • dht11_wait_repon():等待DHT11的响应信号。DHT11会在接收到启动信号后,拉低引脚80μs,然后拉高80μs作为响应。
  • dht11_get_bit():从DHT11读取一个数据位。DHT11通过拉低引脚50μs然后拉高或保持低电平(根据数据位是0还是1)来发送数据位。
  • dht11_get_value():读取DHT11发送的完整数据包(40位),并将其存储在提供的数组中。同时,计算校验和并验证数据的正确性

这篇关于dht11 + Sc32440驱动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1116322

相关文章

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta

驱动(RK3588S)第七课时:单节点设备树

目录 需求一、设备树的概念1、设备树的后缀名:2、设备树的语法格式3、设备树的属性(重要)4、设备树格式举例 二、设备树所用函数1、如何在内核层种获取设备树节点:2、从设备树上获取 gpio 口的属性3、获取节点上的属性只针对于字符串属性的4、函数读取 np 结点中的 propname 属性的值,并将读取到的 u32 类型的值保存在 out_value 指向的内存中,函数的返回值表示读取到的

驱动安装注册表指令

HKCR: HKEY_CLASSES_ROOT HKCU: HKEY_CURRENT_USER HKLM: HKEY_LOCAL_MACHINE HKU: HEKY_USER HER: 相对根键

UMDF驱动安装

VS2013 + WDF8.1,UMDF驱动选择User Mode Driver,不要选User Mode Driver 2.0,否则Win7安装有问题,如图 另外,在驱动安装时不要忘记WUDFUpdate_<主版本号><次版本号>.dll文件,具体文件名在INF中查找。此文件可在WDF的安装目录中找到。注意:在WDF的安装目录中会有3个WUDFUpdate_xxx.dll文件,x86,x6

电脑驱动分类

电脑驱动程序(驱动程序)是操作系统与硬件设备之间的桥梁,用于使操作系统能够识别并与硬件设备进行通信。以下是常见的驱动分类: 1. 设备驱动程序 显示驱动程序:控制显卡和显示器的显示功能,负责图形渲染和屏幕显示。 示例:NVIDIA、AMD 显示驱动程序。打印机驱动程序:允许操作系统与打印机通信,控制打印任务。 示例:HP、Canon 打印机驱动程序。声卡驱动程序:管理音频输入和输出,与声卡硬件

麒麟系统安装GPU驱动

1.nvidia 1.1显卡驱动 本机显卡型号:nvidia rtx 3090 1.1.1下载驱动 打开 https://www.nvidia.cn/geforce/drivers/ 也可以直接使用下面这个地址下载 https://www.nvidia.com/download/driverResults.aspx/205464/en-us/ 1.1.3安装驱动 右击,

windows10 卸载网络驱动以及重新安装

右键桌面此电脑的图标,点击管理,设备管理器—网络适配器,找到下图中的驱动(不同的系统或者显卡会导致网卡驱动名称与下图不一样,多为Realtek开头),右键选择卸载设备,然后重启电脑,系统会自动重新安装驱动 新电脑首次安装驱动: 根据主板厂家,比如华硕,进入华硕官网,点击服务支持,点击下载中心,选择型号,点击右侧驱动程序和工具软件,选择windows版本,下载相应的驱动,下载完之后在对应文件中找

笔记整理—内核!启动!—kernel部分(1)驱动与内核的关系

首先,恭喜完成了uboot部分的内容整理,其次补充一点,uboot第一部分和第二部分的工作不是一定的,在不同的版本中,可能这个初始化早一点,那个的又放在了第二部分,版本不同,造成的工作顺序不同,但终归是要完成基本内容初始化并传参给kernel的。         那么至于驱动与内核的关系,用一张图来说明最适合不过:         驱动位于OS层的中下层与硬件相接。驱动是内

读源码笔记--文件过滤驱动FileSpy第1篇 -- DriverEntry

今天只读FileSpy的DriverEntry,位于源文件:filespy.c。 // // 全局变量. // ULONG gFileSpyDebugLevel = DEFAULT_FILESPY_DEBUG_LEVEL; #if WINVER >= 0x0501 ULONG gFileSpyAttachMode = FILESPY_ATTACH_ALL_VOLUMES; #else ULON

Circuit Design 三极管驱动蜂鸣器电路 及 蜂鸣器两端电压正确但是不响的解决方案

利用三极管进行电流放大的蜂鸣器驱动电路图: (百度图片找的) 我用有源蜂鸣器实现的这个电路,但是蜂鸣器不响。 details: 1. VCC =5V 蜂鸣器两端的直接电压约为4.5V, 但是蜂鸣器不响。 2. 将蜂鸣器直接接在4.5V的电源两端,蜂鸣器响。(说明蜂鸣器是好的) 3. 测了三极管各个管脚的电压, 和理论上的是一致的。 情况很奇怪,换了好几个三极管结果都是一样的,