文章目录
- 前言
- 一、DTH11 模块介绍
- 二、设备树设置
- 三、驱动程序
- 四、测试程序
- 五、上机测试及效果
- 总结
前言
DHT11 是一款可测量 温度 和 湿度 的传感器。比如市面上一些空气加湿器,会测量空气中湿度,再根据测量结果决定是否继续加湿。
一、DTH11 模块介绍
- DHT11通信过程:
主机通过一条数据线与DH11连接,主机通过这条线发命令给DHT11,DHT11再通过这条线把数据发送给主机。
当主机没有与DHT11通信时,总线处于空闲状态,此时总线电平由于上拉电阻的作用处于高电平。
- 主机将对应的GPIO管脚配置为输出,准备向DHT11发送数据;
- 主机发送一个开始信号:
开始信号 = 一个低脉冲 + 一个高脉冲。低脉冲至少持续18ms,高脉冲持续20-40us。 - 主机将对应的GPIO管脚配置为输入,准备接受DHT11传来的数据,这时信号由上拉电阻拉高;
- DHT11发出响应信号:
响应信号 = 一个低脉冲 + 一个高脉冲。低脉冲持续80us,高脉冲持续80us。 - DHT11发出数据信号:
数据为0的一位信号 = 一个低脉冲 + 一个高脉冲。低脉冲持续50us,高脉冲持续26~28us。
数据为1的一位信号 = 一个低脉冲 + 一个高脉冲。低脉冲持续50us,高脉冲持续70us。
DHT11发出结束信号: - 最后1bit数据传送完毕后,DHT11拉低总线50us,然后释放总线,总线由上拉电阻拉高进入空闲状态。
- 数据格式:
8 bit 湿度整数数据 + 8 bit 湿度小数数据 + 8 bit 温度整数数据 + 8 bit 温度小数数据 + 8 bit 校验和。
(5 字节数据,共 40 位 )
数据传送正确时,校验和等于“8bit湿度整数数据+8bit湿度小数数据+8bi温度整数数据+8bit温度小数数据”所得结果的末8位。
二、设备树设置
设备树 中 compatible 与 驱动程序 进行匹配。
将模块分别接到 开发板的 gpio4-19引脚。每一组 GPIO 有 32 个引脚。
配置设备树需要对 GPIO 引脚 以及相关的 pincontrol 配置。由于本实验是使用 DTH11 模块,所以不需要配置 pincontrol 。
三、驱动程序
- 根据框架编写基本驱动程序:
首先要 定义一个file_operations 结构体,在入口函数里对其 注册,在 出口函数里卸载。 实现辅助信息,使用 class_create 创建类 , device_create 创建设备节点。
定义一个 platform_driver。也是在入口函数里对其 注册,在 出口函数里卸载。
这些基本代码的详细实现可以参考我之前的文章 :SR501人体红外模块
- 实现 probe 函数。
总线处于空闲状态,此时总线电平由于上拉电阻的作用处于高电平。
static int dht11_probe(struct platform_device *pdev)
{/* 获取引脚信息,将其设置为输出引脚高电平 */dht11_data_pin = gpiod_get(&pdev->dev, NULL, GPIOD_OUT_HIGH);if (IS_ERR(dht11_data_pin)){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);}/* 创建设备节点 */device_create(dht11_class, NULL, MKDEV(major, 0), NULL, "mydht11");return 0;
}
- 首先将 gpio 引脚设置为 输出引脚,准备向 DTH11 发送数据。
static void dht11_reset(void)
{gpiod_direction_output(dht11_data_pin, 1);
}
- 然后主机发送一个开始信号。
发送开始 信号完毕,将 引脚设置为 输入引脚,等待 DHT11 发送数据,准备接收数据。
static void dht11_start(void)
{mdelay(30);gpiod_set_value(dht11_data_pin, 0);mdelay(20);gpiod_set_value(dht11_data_pin, 1);udelay(40); gpiod_direction_input(dht11_data_pin); udelay(2);
}
- DHT11 发出响应信号,之后发送数据。
static int dht11_wait_for_ready(void)
{int timeout_us = 20000; //设置超时时间/* 等待低电平 */while (gpiod_get_value(dht11_data_pin) && --timeout_us){udelay(1);}if (!timeout_us){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -1;}/* 现在是低电平,等待高电平 */timeout_us = 200;while (!gpiod_get_value(dht11_data_pin) && --timeout_us){udelay(1);}if (!timeout_us){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -1;}/* 现在是高电平,等待低电平 */timeout_us = 200;while (gpiod_get_value(dht11_data_pin) && --timeout_us){udelay(1);}if (!timeout_us){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -1;}return 0;
}
- 读一字节的数据。
每字节八位数据,按位读取。
怎么判断读出的数据是 0 还是 1 呢?
通过判断高电平的持续时间可 得出写入的数据。当 高电平 持续26~28us,表示输出 0 。高脉冲持续70us,表明数据是 1 .
static int dht11_read_byte(unsigned char *buf)
{int i;unsigned char data = 0;int timeout_us = 200;for (i = 0; i <8; i++){/* 现在是低电平 *//* 等待高电平 */timeout_us = 400;while (!gpiod_get_value(dht11_data_pin) && --timeout_us){udelay(1);}if (!timeout_us){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -1;}/* 现在是高电平 *//* 等待低电平,累加高电平的时间 */timeout_us = 20000000;udelay(40);if (gpiod_get_value(dht11_data_pin)){/* get bit 1 */data = (data << 1) | 1;/* 当前位的高电平未结束, 等待 */timeout_us = 400;while (gpiod_get_value(dht11_data_pin) && --timeout_us){udelay(1);}if (!timeout_us){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -1;}}else{/* get bit 0 */data = (data << 1) | 0;}}*buf = data;return 0;
}
- 读五字节数据。
将 读出的数据放入数组中,传递给应用程序。
static ssize_t dht11_drv_read (struct file *file, char __user *buf, size_t size, loff_t *offset)
{unsigned long flags;int i,err;unsigned char data[5];if (size != 4)return -EINVAL;local_irq_save(flags); // 关中断/* 1. 发送高脉冲启动DHT11 */dht11_reset();dht11_start();/* 2. 等待DHT11就绪 */if (dht11_wait_for_ready()){local_irq_restore(flags); // 恢复中断printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -EAGAIN;}/* 3. 读5字节数据 */for (i = 0; i < 5; i++){if (dht11_read_byte(&data[i])){local_irq_restore(flags); // 恢复中断printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);return -EAGAIN;}}dht11_reset();local_irq_restore(flags); // 恢复中断/* 4. 根据校验码验证数据 */if (data[4] != (data[0] + data[1] + data[2] + data[3])){printk("%s %s line %d\n", __FILE__, __FUNCTION__, __LINE__);//return -1;}/* 5. copy_to_user */ /* data[0]/data[1] : 湿度 *//* data[2]/data[3] : 温度 */err = copy_to_user(buf, data, 4);return 4;
}
四、测试程序
判断参数,打开文件,read 函数读出 测出的温度,湿度。
if (argc != 2) {printf("Usage: %s <dev>\n", argv[0]);return -1;}fd = open(argv[1], O_RDWR); if (fd == -1){printf("can not open file %s\n", argv[1]);return -1;}while (1){if (read(fd, data, 4) == 4){printf("get humidity : %d.%d\n", data[0], data[1]);printf("get temprature: %d.%d\n", data[2], data[3]);}else {printf("get humidity/temprature: -1\n");}sleep(5); // 等待 5 秒}close(fd);
五、上机测试及效果
执行 insmod
命令可以将 .ko 文件加载到内核中,再 执行测试程序。(rmmod
命令可以卸载已加载的模块,lsmod
命令 可以观察已加载到内核的文件。)
/dev/mydht11 是 驱动程序中创建的设备节点( device_create )。