Linux驱动之DHT11温湿度传感器

DHT11驱动开发

一、开发环境

ITOP4412开发板
linux4.14

二、DHT11基本信息

1、传感器性能说明

在这里插入图片描述

2、接口说明

在这里插入图片描述

3、电源引脚

DHT11的供电电压为3-5.5V。传感器上电后,要等待1s 以越过不稳定状态在此
期间无需发送任何指令。电源引脚(VDD,GND)之间可增加一个100nF 的电容,用以去
耦滤波。

4、数据格式

一次完整的数据传输为40bit,高位先出。
1、采集数据格式:
8bit湿度整数数据+8bit湿度小数数据
+8bi温度整数数据+8bit温度小数数据
+8bit校验和
小数部分用于以后扩展,现在读出为0, 也就是小数部分是不需要使用的

2、数据校验
数据传送正确时校验和数据等于
“8bit湿度整数数据+8bit湿度小数数据+8bi温度整数数据+8bit温度小数数据”所得结果的末8位。

5、通信过程

用户MCU发送一次开始信号后,DHT11从低功耗模式转换到高速模式,等待主
机开始信号结束后
,DHT11发送响应信号,送出40bit的数据,并触发一次信号采集,
用户可选择读取部分数据.从模式下,DHT11接收到开始信号触发一次温湿度采集,
如果没有接收到主机发送开始信号,DHT11不会主动进行温湿度采集.采集数据后
转换到低速模式。

在这里插入图片描述
总线空闲状态为高电平,主机把总线拉低等待DHT11响应,主机把总线拉低必
须大于18毫秒,保证DHT11能检测到起始信号。
DHT11接收到主机的开始信号后,
等待主机开始信号结束,然后发送80us低电平响应信号.主机发送开始信号结束
后,延时等待20-40us后, 读取DHT11的响应信号
,主机发送开始信号后,可以切换
到输入模式,或者输出高电平均可, 总线由上拉电阻拉高。
在这里插入图片描述
总线为低电平,说明DHT11发送响应信号,DHT11发送响应信号后,再把总线拉
高80us,准备发送数据,**每一bit数据都以50us低电平时隙开始,高电平的长短定
了数据位是0还是1.**格式见下面图示.如果读取响应信号为高电平,则DHT11没有
响应,请检查线路是否连接正常.当最后一bit数据传送完毕后,DHT11拉低总线
50us,随后总线由上拉电阻拉高进入空闲状态。

数据’0’

在这里插入图片描述

数据’1’

在这里插入图片描述

6、电气特性

在这里插入图片描述

三、驱动程序

1、DHT11与ITOP4412硬件连接

在这里插入图片描述

2、设备树

pinctrl子系统中引用DHT11的管脚

exynos4412-pinctrl.dtsi
在这里插入图片描述
exynos4412-itop-elite.dts
在这里插入图片描述

3、驱动程序

/*************************************************************************
    > File Name: DHT11.c
    > 作者:YJK
    > Mail: 745506980@qq.com
    > Created Time: 2021年05月20日 星期四 21时33分51秒
 ************************************************************************/
#include<linux/init.h>
#include<linux/module.h>
#include <linux/platform_device.h>
#include <linux/of_gpio.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/delay.h>  //内核延时 忙等
#include <linux/uaccess.h>

#define ERROR_PRINT(x) do{	\
		printk(x " is error!\n"); \
		return -1; \
	}while(0)

#define DEVICE_NAME "dht11"

#define HIGH 1
#define LOW 0

/*1、platform匹配*/
/*2、匹配成功后,注册字符设备或杂项设备*/

int dht11_GPIO; //DHT11_GPIO编号

struct dht11_info {
	dev_t devt;
	struct cdev cdev;
	struct class *dht11_class;
	struct device *dht11_device;
};
struct dht11_info dht11;
/*起始信号
主机发送采集信号
主机由高电平-->低电平(至少18ms)--->高电平(20-40us)

DHT响应信号 低电平(80us)--->高电平(80us)

数据开始     低电平(50us)
数据0      高电平 26us-28us
数据1    	 高电平70us
*/

void end(void)
{
	gpio_direction_output(dht11_GPIO, HIGH);
//	gpio_set_value(dht11_GPIO, HIGH);
}
/*数据采集*/
unsigned char get_dht11_value(void)
{
	int i;
	unsigned char data = 0;
	for (i = 0; i < 8; i++) //8bit
	{
		if (gpio_get_value(dht11_GPIO) == LOW)
		{
			while(gpio_get_value(dht11_GPIO) == LOW); //低电平数据起始
			/*当变为高电平时, 等待40us 查看是否为高电平
				如果为高电平 则为 1
				如果为低电平 则为 0
			*/
		//	printk("LOW END\n");
			udelay(30);
			if (gpio_get_value(dht11_GPIO) == HIGH) //1
			{
				data |= (0x1 << (7 - i));  //数据自高位到低位
				udelay(40);
			}
			else
				data &= ~(0x1 << (7 - i)); // 置为0
		}
		else
			printk("------------ HIGH");
	}
	return data;
}

ssize_t dht11_read(struct file *file, char __user * user_buf, size_t size, loff_t * loff)
{
	unsigned char dht11_data[5];
	printk("start\n");
	ssleep(1);
	gpio_set_value(dht11_GPIO, LOW);
	mdelay(20); //拉低20ms
	/*拉高20-40us*/
	gpio_set_value(dht11_GPIO, HIGH);
	udelay(30); //拉高30us
	//设置为输入
	gpio_direction_input(dht11_GPIO);
	/*DHT11响应信号*/
	if (gpio_get_value(dht11_GPIO) == LOW)
	{
	//	printk("xiangying\n");
		//80us 低电平
		while(gpio_get_value(dht11_GPIO) == LOW);
		//80us 高电平
		while(gpio_get_value(dht11_GPIO) == HIGH);
		dht11_data[0] = get_dht11_value();	//湿度整数数据
		dht11_data[1] = get_dht11_value();	//湿度小数数据  0
		dht11_data[2] = get_dht11_value();	//温度整数数据
		dht11_data[3] = get_dht11_value();	//温度小数数据
		dht11_data[4] = get_dht11_value();	//校验和
		printk("data end\n");
	}
	else
		printk("start is error!\n");

	if (copy_to_user(user_buf, dht11_data, size) != 0)
	{
		ERROR_PRINT("copy_to_user");
	}
	printk("copy to user\n");
	//恢复为输出模式高电平
	end();
	return 0;
}
int dht11_open(struct inode * inode, struct file * file)
{
	return 0;
}
int dht11_release(struct inode * inode, struct file *file)
{
	return 0;
}
struct file_operations fops = {
	.open = dht11_open,
	.release = dht11_release,
	.read = dht11_read,
	.owner = THIS_MODULE
};

/*匹配成功进入probe函数*/
int dht11_probe(struct platform_device *pdev)
{
	/*匹配成功之后,获取硬件资源,然后注册字符设备*/
	/*SHT11-gpios = <&gpk3 0 GPIO_ACTIVE_HIGH>;*/
	int ret;
	printk("probe\n");
	dht11_GPIO = of_get_named_gpio(pdev->dev.of_node, "SHT11-gpios", 0);
	if (dht11_GPIO < 0)
	{
		ERROR_PRINT("of_get_named_gpio");
	}
	/*获取资源*/
	ret = gpio_request(dht11_GPIO, "dht11_gpio");
	if (ret != 0)
	{
		ERROR_PRINT("gpio_request!");
	}
	ret = gpio_direction_output(dht11_GPIO, HIGH);
	if (ret != 0)
	{
		ERROR_PRINT("gpio_direction_output");
	}
	/*注册字符设备驱动*/
	/*分配设备号*/
	ret = alloc_chrdev_region(&dht11.devt, 0, 1, "dht11");
	if (ret != 0)
	{
		ERROR_PRINT("alloc_chrdev_region");
	}
	/*初始化cdev*/
	dht11.cdev.owner = THIS_MODULE;
	cdev_init(&dht11.cdev, &fops);

	/*将cdev注册到内核*/
	ret = cdev_add(&dht11.cdev, dht11.devt,1);
	if (ret != 0)
	{
		ERROR_PRINT("cdev_add");
	}
	/*创建设备class*/
	dht11.dht11_class = class_create(THIS_MODULE, "dht11_class");
	if (dht11.dht11_class == NULL)
	{
		ERROR_PRINT("class_create");
	}
	/*在class下创建设备节点*/
	dht11.dht11_device = device_create(dht11.dht11_class, NULL, dht11.devt, NULL, DEVICE_NAME);
	if (dht11.dht11_device == NULL)
	{
		ERROR_PRINT("device_create");
	}
	printk("probe success!\n");
	return 0;
}
int dht11_remove(struct platform_device *pdev)
{
	return 0;
}
struct of_device_id of_match_table[] = {
	{.compatible = "dht11"},
	{}
}; //设备树匹配

struct platform_driver pdrv = {
	.probe = dht11_probe,
	.remove = dht11_remove,
	.driver = {
		.name = "dht11",
		.owner = THIS_MODULE,
		.of_match_table = of_match_table
	},
};
static int dht11_init(void)
{
	//注册platform_driver
	int ret = platform_driver_register(&pdrv);
	if (ret != 0)
		ERROR_PRINT("platform driver register");

	return 0;
}
static void dht11_exit(void)
{
	/*注销设备节点*/
	device_destroy(dht11.dht11_class, dht11.devt);
	/*注销class*/
	class_destroy(dht11.dht11_class);
	/*注销字符设备*/
	cdev_del(&dht11.cdev);
	/*注销设备号*/
	unregister_chrdev_region(dht11.devt, 1);
	/*注销GPIO*/
	gpio_free(dht11_GPIO);
	/*注销设备节点*/
	/*注销platform_driver*/
	platform_driver_unregister(&pdrv);
	printk("bye bye\n");
}

module_init(dht11_init);
module_exit(dht11_exit);

MODULE_LICENSE("GPL");

4、应用程序

/*************************************************************************
    > File Name: app_dht11.c
    > 作者:YJK
    > Mail: 745506980@qq.com
    > Created Time: 2021年05月21日 星期五 12时29分13秒
 ************************************************************************/

#include<stdio.h>
#include <fcntl.h>
#include <unistd.h>
int main(int argc,char *argv[])
{
	unsigned char buf[5] = {0};
	int fd = open("/dev/dht11", O_RDWR);
	if (fd == -1)
	{
		perror("open");
		return -1;
	}
	printf("open success!\n");
	while(1)
	{
		int ret = read(fd, buf, 5);
		if (ret == -1)
		{
			perror("read");
			return -1;
		}
		/*校验和*/
		int crc = buf[0] + buf[1] + buf[2] + buf[3];
		int i = 0;
		if (crc & 0xFF == buf[4])
		{
			printf("humidity %d% \n", buf[0]);
			printf("temperatuere %dC\n", buf[2]);
		}
		sleep(2);
	}

	close(fd);
	return 0;
}

在这里插入图片描述
OVER!
DHT11 这个传感器相对来说还是比较简单的,模拟时序即可
1、主机发送起始信号 高电平---->低电平(至少18ms)----->高电平(20-40us)
2、DHT11应答 DHT11 高电平---->低电平(80us)------->高电平(80us)
3、数据传输 开始数据传输信号 高电平----->(1Bit起始)低电平(50us) ---->高电平(26us-28us数据0 (70us 数据1))
数据传输完毕 DHT11 拉低总线50us 然后主机将总线拉高

  • 11
    点赞
  • 65
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Linux图像传感器驱动是指在Linux操作系统下为图像传感器设计和开发的驱动程序。图像传感器是一种将光信号转换为数字图像的设备,被广泛用于数码相机、手机摄像头、工业视觉等领域。 Linux图像传感器驱动的主要功能包括对图像传感器进行初始化、配置和控制,以确保传感器能够正常工作和正常获取图像数据。Linux操作系统为各种硬件设备提供了统一的接口和驱动框架,使得开发者能够方便地编写并调试硬件驱动程序。 在设计和开发Linux图像传感器驱动时,首先需要对具体的图像传感器进行了解,并了解其硬件规范和特性。然后,根据传感器的接口和通信协议,编写相应的驱动程序代码。常见的图像传感器接口包括MIPI CSI-2(移动设备图像接口)和ISP(图像信号处理器)接口。 在驱动程序中,需要实现图像传感器的初始化、配置参数的读写、图像数据的采集和处理等功能。此外,还需要考虑和适配Linux操作系统的驱动框架,与其他子系统和设备进行交互和通信,如视频子系统V4L2(Video for Linux 2)。 Linux图像传感器驱动的编写需要具备扎实的嵌入式系统和Linux操作系统的基础知识,熟悉相关的硬件接口和通信协议,以及掌握C/C++等编程语言。同时,驱动程序的设计和开发还需要经过充分的测试和调试,确保其稳定可靠、高效运行。 总之,Linux图像传感器驱动为图像传感器的正常工作提供了必要的支持,在嵌入式系统和数字图像处理中具有重要的意义和应用价值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值