资源简介
说明:通过内核驱动dht11实现温度湿度的检测
核心代码:
#include <linux/module.h>
#include <linux/delay.h>
#include <linux/sched.h>
#include <linux/fs.h>
#include <linux/interrupt.h>
#include <linux/init.h>
#include <linux/miscdevice.h>
#include <linux/wait.h>
#include <linux/gpio.h>
#include <linux/io.h>
#include <linux/uaccess.h>
/*定义DHT11接入的IO口*/
#define DHT11_PIN EXYNOS4_GPB(4)
#define DEVNAME "dht11"
/* 用于接受采集到的数据*/
unsigned char dht11_data_buf[6];
/* 判断校验的标记*/
unsigned char check_flag;
/* read one bit */
static int dht11_read_bit(void)
{
gpio_direction_input(DHT11_PIN);
return gpio_get_value(DHT11_PIN);
}
/* read byte */
static unsigned char dht11_read_byte(void)
{
int i = 0;
int num ;
unsigned char flag = 0;
unsigned char data = 0;
for(num = 0; num < 8; num ){
i = 0;
/*没一个位发送起始的时候,会有50us低电平*/
while(gpio_get_value(DHT11_PIN) == 0){
udelay(10);
i ;
if(i > 5){
break;
}
}
flag = 0x00;
/*26-28 us 高电平表示'0'*/
udelay(30);
if(gpio_get_value(DHT11_PIN)){
flag = 0x01;
}
i = 0;
/*如果读到的数据为'1'的话,70us 高电平,将这段时间的高电平读取完毕*/
while(gpio_get_value(DHT11_PIN)){
udelay(20);
i ;
if(i > 3)
break;
}
data <<= 1;
data |= flag;
}
return data;
}
/* 设备 read_data*/
static void dht11_read_data(void)
{
int i = 0;
/*
主机把总线拉低必须大于18毫秒,保证DHT11能检测到起始信号。
DHT11接收到主机的开始信号后,等待主机开始信号结束,然后发送80us低电平响应信号.
主机发送开始信号结束后,延时等待20-40us后, 读取DHT11的响应信号,
主机发送开始信号后,可以切换到输入模式,或者输出高电平均可, 总线由上拉电阻拉高。
*/
gpio_direction_output(DHT11_PIN,0);
mdelay(30);
gpio_direction_output(DHT11_PIN,1);
udelay(20);
if(dht11_read_bit() == 0){
while(!gpio_get_value(DHT11_PIN)){
udelay(10);
i ;
if(i > 8){
printk("dht11 read data %d error\n",__LINE__);
break;
}
}
i = 0;
while(gpio_get_value(DHT11_PIN)){
udelay(10);
i ;
if(i > 8){
printk("dht11 read data %d error\n",__LINE__);
break;
}
}
for(i = 0; i < 5; i ){
dht11_data_buf[i] = dht11_read_byte();
}
dht11_data_buf[5] = dht11_data_buf[0] dht11_data_buf[1]
dht11_data_buf[2] dht11_data_buf[3];
if(dht11_data_buf[5] == dht11_data_buf[4]){
check_flag = 0xff;
}else if(dht11_data_buf[4] != dht11_data_buf[5]){
check_flag = 0x00;
}
}
}
/* 设备 read*/
static ssize_t dht11_read(struct file *file, char __user *buffer, ssize_t size, loff_t *offset)
{
int ret;
// unsigned long flags;
// local_irq_save(flags);
dht11_read_data();
// local_irq_restore(flags);
if(check_flag == 0xff){
ret = copy_to_user(buffer,dht11_data_buf,sizeof(dht11_data_buf));
if(ret < 0){
printk("copy_to_user error\n");
return -EAGAIN;
}else{
return 0;
}
}else{
printk("check error\n");
return -EAGAIN;
}
}
/* 定义杂项设备操作的结构体并且初始化成员的函数指针 */
static const struct file_operations dht11_fops = {
.read = dht11_read,
};
/* 定义杂项设备结构体*/
static struct miscdevice dht11_miscdev = {
.minor = MISC_DYNAMIC_MINOR, /* 自动分配次设备号 */
.name = DEVNAME, /* 杂项设备的名字 */
.fops = &dht11_fops, /* 杂项设备操作的结构体(驱动函数集合)*/
};
static int dht11_init(void)
{
int ret;
ret = gpio_request(DHT11_PIN,DEVNAME);
if(ret){
printk("gpio_retquest for dht11 is filed\n");
return ret;
}
ret = misc_register(&dht11_miscdev);
return ret;
}
static void dht11_exit(void)
{
misc_deregister(&dht11_miscdev);
gpio_free(DHT11_PIN);
}
module_init(dht11_init);
module_exit(dht11_exit);
MODULE_LICENSE("GPL");
代码片段和文件信息
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
/*定义DHT11接入的IO口*/
#define DHT11_PIN EXYNOS4_GPB(4)
#define DEVNAME “dht11“
/* 用于接受采集到的数据*/
unsigned char dht11_data_buf[6];
/* 判断校验的标记*/
unsigned char check_flag;
/* read one bit */
static int dht11_read_bit(void)
{
gpio_direction_input(DHT11_PIN);
return gpio_get_value(DHT11_PIN);
}
/* read byte */
static unsigned char dht11_read_byte(void)
{
int i = 0;
int num ;
unsigned char flag = 0;
unsigned char data = 0;
for(num = 0; num < 8; num ++){
i = 0;
/*没一个位发送起始的时候,会有50us低电平*/
while(gpio_get_value(DHT11_PIN) == 0){
udelay(10);
i++ ;
if(i > 5){
break;
}
}
flag = 0x00;
/*26-28 us 高电平表示‘0‘*/
udelay(30);
if(gpio_get_value(DHT11_PIN)){
flag = 0x01;
}
i = 0;
/*
相关资源
- Linux优先级时间片调度C++源码
- 获取linux内核核心信息(shell脚本)
- Linux内核代码
- linux下获取CPU内存使用信息,网络流量
- 非阻塞式AT指令发送接收工程,可驱动
- CS5532 的 STM32 驱动程序
- Linux驱动,SPI驱动
- 灯珠内带驱动IC 的彩灯条控制!
- AD7276驱动程序 STM32
- [野火]《RT-Thread 内核实现与应用开发
- 伺服电机驱动
- STM32_HI2C_OLED硬件方式驱动OLED
- linux驱动原子操作
- S32K14X芯片LIN驱动
- 步进电机驱动程序(51单片机可编程电
- bk 1080 收音机驱动(fm_BK1080.c)
- [野火®]《FreeRTOS 内核实现与应用开
- 海思平台Hi3559A的pwm_led驱动
- android和linux平台下的nanocom源码
- 工业相机通讯驱动控件,用于以太协
- 单片机Futaba VFD显示屏驱动
- linux内核的裁剪和方法
- Linux网络编程.pdf(入门级教程)
- Linux TCP IP 协议栈分析.pdf
- opengl:基于linux下雷达ppi
- C++ 获取驱动器的卷标(提高篇-001)
- C++ 映射网络驱动器(提高篇-529).z
- linux ymodem串口发送
- OLEDstm32驱动(0.96OLED显示屏STM32F103RC
- 直流电机调速驱动
评论
共有 条评论