基于ARM-LINUX的温度传感器驱动(DS18B20) .
DS18B20数字温度传感器接线方便,封装成后可应用于多种场合,如管道式,螺纹式,磁铁吸附式,不锈钢封装式,型号多种多样,有LTM8877,LTM8874等等。主要根据应用场合的不同而改变其外观。封装后的DS18B20可用于电缆沟测温,高炉水循环测温,锅炉测温,机房测温,农业大棚测温,洁净室测温,弹药库测温等各种非极限温度场合。耐磨耐碰,体积小,使用方便,封装形式多样,适用于各种狭小空间设备数字测温和控制领域。
技术性能描述
1. 独特的单线接口方式,DS18B20在与微处理器连接时仅需要一条口线即可实现微处理器与DS18B20的双向通讯。
2. 测温范围 -55℃~+125℃,固有测温分辨率0.5℃。
3. 支持多点组网功能,多个DS18B20可以并联在唯一的三线上。
4. 工作电源: 3~5V/DC
5. 在使用中不需要任何外围元件
6. 测量结果以9~12位数字量方式串行传送
DS18b20封装
DS18B20 引脚功能: GND 电压地 •DQ 单数据总线 •VDD 电源电压 •NC 空引脚
DS18b20与处理器的连接
DS18B20 工作原理及应用
DS18B20 的温度检测与数字数据输出全集成于一个芯片之上,从而抗干扰力更强。其一个工作周期可分为两个部分,即温度检测和数据处理。在讲解其工作流程之前我们有必要了解 18B20的内部存储器资源。18B20 共有三种形态的存储器资源。它们分别是:
ROM 只读存储器:
用于存放 DS18B20ID 编码,其前 8 位是单线系列编码(DS18B20 的编码是19H) ,后面48 位是芯片唯一的序列号,最后 8位是以上 56的位的 CRC码(冗余校验)。数据在出产时设置不由用户更改。DS18B20 共 64 位 ROM。
RAM 数据暂存器:
用于内部计算和数据存取,数据在掉电后丢失,DS18B20 共9 个字节 RAM,每个字节为 8 位。第1、2 个字节是温度转换后的数据值信息,第 3、4 个字节是用户 EEPROM(常用于温度报警值储存)的镜像。在上电复位时其值将被刷新。第 5 个字节则是用户第 3 个 EEPROM的镜像。第 6、7、8 个字节为计数寄存器,是为了让用户得到更高的温度分辨率而设计的,同样也是内部温度转换、计算的暂存单元。第 9 个字节为前 8个字节的 CRC码。EEPROM 非易失性记忆体,用于存放长期需要保存的数据,上下限温度报警值和校验数据, DS18B20共3位EEPROM,并在 RAM 都存在镜像,以方便用户操作。
控制器对 18B20 操作流程:
1、 复位:首先我们必须对 DS18B20 芯片进行复位,复位就是由控制器(单片机)给 DS18B20单总线至少 480uS 的低电平信号。当 18B20 接到此复位信号后则会在 15~60uS 后回发一个芯片的存在脉冲。
2、 存在脉冲:在复位电平结束之后,控制器应该将数据单总线拉高,以便于在 15~60uS 后接收存在脉冲,存在脉冲为一个 60~240uS 的低电平信号。至此,通信双方已经达成了基本的协议,接下来将会是控制器与 18B20 间的数据通信。如果复位低电平的时间不足或是单总线的电路断路都不会接到存在脉冲,在设计时要注意意外情况的处理。
3、 控制器发送 ROM 指令:双方打完了招呼之后最要将进行交流了,ROM 指令共有 5条,每一个工作周期只能发一条,ROM指令分别是读 ROM 数据、指定匹配芯片、跳跃 ROM、芯片搜索、报警芯片搜索。ROM 指令为 8 位长度,功能是对片内的 64位光刻 ROM进行操作。其主要目的是为了分辨一条总线上挂接的多个器件并作处理。诚然,单总线上可以同时挂接多个器件,并通过每个器件上所独有的 ID号来区别,一般只挂接单个 18B20芯片时可以跳过 ROM 指令(注意:此处指的跳过 ROM指令并非不发送 ROM 指令,而是用特有的一条“跳过指令” )
4、 控制器发送存储器操作指令:在 ROM 指令发送给 18B20 之后,紧接着(不间断)就是发送存储器操作指令了。操作指令同样为 8 位,共 6 条,存储器操作指令分别是写 RAM 数据、读RAM 数据、将 RAM 数据复制到 EEPROM、温度转换、将 EEPROM中的报警值复制到 RAM、工作方式切换。存储器操作指令的功能是命令 18B20 作什么样的工作,是芯片控制的关键。
5、 执行或数据读写:一个存储器操作指令结束后则将进行指令执行或数据的读写,这个操作要视存储器操作指令而定。如执行温度转换指令则控制器(单片机)必须等待 18B20 执行其指令,一般转换时间为 500uS。如执行数据读写指令则需要严格遵循 18B20 的读写时序来操作。
若要读出当前的温度数据我们需要执行两次工作周期,第一个周期为复位、跳过 ROM 指令、执行温度转换存储器操作指令、等待 500uS 温度转换时间。紧接着执行第二个周期为复位、跳过 ROM指令、执行读 RAM 的存储器操作指令、读数据(最多为 9 个字节,中途可停止,只读简单温度值则读前 2 个字节即可)。其它的操作流程也大同小异,在此不多介绍。关于ds18b20的数据手册网上资源较为丰富,这里不再详细介绍,下面是基于ARM-LINUX的驱动程序,在arm-gcc 编译后测试通过。
- #include <linux/module.h>
- #include <linux/moduleparam.h>
- #include <linux/fs.h>
- #include <linux/proc_fs.h>
- #include <linux/errno.h>
- #include <linux/cdev.h>
- #include <linux/ioctl.h>
- #include <linux/fcntl.h>
- #include <linux/sched.h>
- #include <linux/poll.h>
- #include <linux/version.h>
- #include <linux/kernel.h>
- #include <linux/init.h>
- #include <linux/device.h>
- #include <linux/mm.h>
- #include <linux/spinlock.h>
- #include <plat/map.h>
- #include <asm/io.h>
- #include <asm/delay.h>
- #include <asm/memory.h>
- #include <asm/uaccess.h>
- #include <asm/mach/map.h>
- #include <asm/uaccess.h>
- #include <mach/regs-gpio.h>
- #include <mach/map.h>
- MODULE_LICENSE("GPL");
- #define GPHCON (*(volatile unsigned int *)S3C2410_GPHCON)
- #define GPHDAT (*(volatile unsigned int *)S3C2410_GPHDAT)
- #define GPHUP (*(volatile unsigned int *)S3C2410_GPHUP)
- static int ds18b20_major = 230; /*静态申请设备号*/
- struct cdev cdev;
- struct class *my_class;
- spinlock_t lock;
- dev_t dev = 0;
- int number_of_devices = 1;
- /*配置为输入模式*/
- void set_conIN(void)
- {
- GPHCON &= ~(1<<19);
- GPHCON &= ~(1<<18);
- }
- /*配置为输出模式*/
- void set_conOUT(void)
- {
- GPHCON |= (1<<18);
- GPHCON &= ~(1<<19);
- }
- /*引脚置位*/
- void set_data(int i)
- {
- if( i == 0 ){
- GPHDAT &= ~(1<<9);
- }else if( i == 1 ){
- GPHDAT |= (1<<9);
- }
- }
- /*复位ds18b20*/
- unsigned int reset_ds18b20(void)
- {
- unsigned int retValue;
- set_conOUT();
- set_data(1);
- __udelay(1);
- set_data(0);
- __udelay(600);
- set_data(1);
- __udelay(20);
- set_conIN();
- __udelay(100);
- /*稍做延时后 如果x=0则初始化成功
- x=1则初始化失*/
- retValue = (GPHDAT >> 9) & 0x01;
- printk("init is %d\n",retValue);
- return retValue;
- }
- /*读取一位温度*/
- unsigned int read_bit(void)
- {
- spin_lock(&lock);
- set_conOUT();
- //set_data(1);
- //__udelay(2);
- set_data(0);
- __udelay(2);
- set_conIN();
- __udelay(1);
- spin_unlock(&lock);
- return ((GPHDAT >> 9) & 0x01);
- }
- /*写一位命令*/
- void write_bit(char bitValue)
- {
- spin_lock(&lock);
- set_conOUT();
- set_data(0);
- __udelay(15);
- if( bitValue == 1 ){
- set_data(1);
- }else{
- set_data(0);
- }
- spin_unlock(&lock);
- __udelay(45);
- set_conIN();
- __udelay(2);
- }
- /*写命令*/
- void write_cmd(char cmd)
- {
- unsigned char i;
- unsigned char temp;
- for(i=0; i<8;i++){
- temp = cmd>>i;
- temp &= 0x01;
- write_bit(temp);
- }
- //__udelay(10);
- }
- /*打开设备*/
- static int ds18b20_open(struct inode *inode,struct file *filp)
- {
- printk (KERN_INFO "HEY! device opened\n");
- //GPHUP &= ~(1<<9);
- GPHUP |= (1<<9);
- spin_lock_init(&lock);
- return 0;
- }
- /*读取数据*/
- static int ds18b20_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)
- {
- char lowValue=0,highValue=0;
- unsigned int i;
- //float value;
- if(reset_ds18b20()){
- printk("init error\n");
- }
- __udelay(400);
- set_conOUT();
- set_data(1);
- write_cmd(0xCC);
- write_cmd(0x44);
- __udelay(100000);
- if(reset_ds18b20()){
- printk("init error\n");
- }
- __udelay(400);
- set_conOUT();
- set_data(1);
- write_cmd(0xcc);
- write_cmd(0xBE);
- /*读取温度转化数值*/
- for(i=0; i<8; i++){
- if( read_bit() ){
- lowValue |= (0x01<<i);
- }
- __udelay(62);
- }
- printk("lowValue is %d\n",lowValue);
- for(i=0; i<8; i++){
- if( read_bit() ){
- highValue |= (0x01<<i);
- }
- __udelay(62);
- }
- printk("highValue is %d\n",highValue);
- #if 0
- i = highValue;
- i <<= 8;
- i = i|lowValue;
- value = i*0.0625;
- printk("kernel is %d\n",value);
- #endif
- highValue <<= 4;
- highValue |= ((lowValue&0xf0)>>4) ;
- /*拷贝内核数据到用户空间*/
- copy_to_user(buffer, &highValue, sizeof(highValue));
- return 0;
- }
- /*写命令,在此置空*/
- static int ds18b20_write(struct file *file, const char *buffer, size_t count, loff_t * ppos)
- {
- return 0;
- }
- static int ds18b20_release(struct inode *inode,struct file *filp)
- {
- printk (KERN_INFO "device closed\n");
- return 0;
- }
- static int ds18b20_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg)
- {
- return 0;
- }
- struct file_operations ds18b20_fops ={
- .owner = THIS_MODULE,
- .open = ds18b20_open,
- .read = ds18b20_read,
- .write = ds18b20_write,
- .ioctl = ds18b20_ioctl,
- .release = ds18b20_release,
- };
- static void ds18b20_setup_cdev(void)
- {
- int error,devno = MKDEV(ds18b20_major,0);
- cdev_init(&cdev,&ds18b20_fops);
- cdev.owner = THIS_MODULE;
- cdev.ops = &ds18b20_fops;
- error = cdev_add(&cdev,devno,1);
- if( error )
- printk(KERN_INFO"Error %d adding ds18b20 %d\n",error,0);
- my_class = class_create(THIS_MODULE,"my_class");
- if(IS_ERR(my_class))
- {
- printk("Err: failed in creating class.\n");
- return;
- }
- device_create(my_class,NULL,devno,NULL,"ds18b20");
- }
- /*注册设备*/
- static int ds18b20_init(void)
- {
- int result;
- dev = MKDEV(ds18b20_major,0);
- if(ds18b20_major)
- result = register_chrdev_region(dev,1,"ds18b20");
- else{
- result = alloc_chrdev_region(&dev,0,1,"ds18b20");
- ds18b20_major=MAJOR(dev);
- }
- if( result < 0 ){
- printk(KERN_WARNING"ds18b20:unable to get major %d\n",ds18b20_major);
- return result;
- }
- if(ds18b20_major == 0 )
- ds18b20_major = result;
- ds18b20_setup_cdev();
- printk("ds18b20 initialized.\n");
- return 0;
- }
- static void __exit ds18b20_exit(void)
- {
- dev_t devno = MKDEV (ds18b20_major, 0);
- device_destroy(my_class,devno);
- class_destroy(my_class);
- cdev_del (&cdev);
- unregister_chrdev_region (devno, number_of_devices);
- printk("ds18b20_major=%d\n",ds18b20_major);
- printk("ds18b20 device uninstalled\n");
- }
- module_init(ds18b20_init);
- module_exit(ds18b20_exit);
#include <linux/module.h> #include <linux/moduleparam.h> #include <linux/fs.h> #include <linux/proc_fs.h> #include <linux/errno.h> #include <linux/cdev.h> #include <linux/ioctl.h> #include <linux/fcntl.h> #include <linux/sched.h> #include <linux/poll.h> #include <linux/version.h> #include <linux/kernel.h> #include <linux/init.h> #include <linux/device.h> #include <linux/mm.h> #include <linux/spinlock.h> #include <plat/map.h> #include <asm/io.h> #include <asm/delay.h> #include <asm/memory.h> #include <asm/uaccess.h> #include <asm/mach/map.h> #include <asm/uaccess.h> #include <mach/regs-gpio.h> #include <mach/map.h> MODULE_LICENSE("GPL"); #define GPHCON (*(volatile unsigned int *)S3C2410_GPHCON) #define GPHDAT (*(volatile unsigned int *)S3C2410_GPHDAT) #define GPHUP (*(volatile unsigned int *)S3C2410_GPHUP) static int ds18b20_major = 230; /*静态申请设备号*/ struct cdev cdev; struct class *my_class; spinlock_t lock; dev_t dev = 0; int number_of_devices = 1; /*配置为输入模式*/ void set_conIN(void) { GPHCON &= ~(1<<19); GPHCON &= ~(1<<18); } /*配置为输出模式*/ void set_conOUT(void) { GPHCON |= (1<<18); GPHCON &= ~(1<<19); } /*引脚置位*/ void set_data(int i) { if( i == 0 ){ GPHDAT &= ~(1<<9); }else if( i == 1 ){ GPHDAT |= (1<<9); } } /*复位ds18b20*/ unsigned int reset_ds18b20(void) { unsigned int retValue; set_conOUT(); set_data(1); __udelay(1); set_data(0); __udelay(600); set_data(1); __udelay(20); set_conIN(); __udelay(100); /*稍做延时后 如果x=0则初始化成功 x=1则初始化失*/ retValue = (GPHDAT >> 9) & 0x01; printk("init is %d\n",retValue); return retValue; } /*读取一位温度*/ unsigned int read_bit(void) { spin_lock(&lock); set_conOUT(); //set_data(1); //__udelay(2); set_data(0); __udelay(2); set_conIN(); __udelay(1); spin_unlock(&lock); return ((GPHDAT >> 9) & 0x01); } /*写一位命令*/ void write_bit(char bitValue) { spin_lock(&lock); set_conOUT(); set_data(0); __udelay(15); if( bitValue == 1 ){ set_data(1); }else{ set_data(0); } spin_unlock(&lock); __udelay(45); set_conIN(); __udelay(2); } /*写命令*/ void write_cmd(char cmd) { unsigned char i; unsigned char temp; for(i=0; i<8;i++){ temp = cmd>>i; temp &= 0x01; write_bit(temp); } //__udelay(10); } /*打开设备*/ static int ds18b20_open(struct inode *inode,struct file *filp) { printk (KERN_INFO "HEY! device opened\n"); //GPHUP &= ~(1<<9); GPHUP |= (1<<9); spin_lock_init(&lock); return 0; } /*读取数据*/ static int ds18b20_read(struct file *filp, char *buffer, size_t count, loff_t *ppos) { char lowValue=0,highValue=0; unsigned int i; //float value; if(reset_ds18b20()){ printk("init error\n"); } __udelay(400); set_conOUT(); set_data(1); write_cmd(0xCC); write_cmd(0x44); __udelay(100000); if(reset_ds18b20()){ printk("init error\n"); } __udelay(400); set_conOUT(); set_data(1); write_cmd(0xcc); write_cmd(0xBE); /*读取温度转化数值*/ for(i=0; i<8; i++){ if( read_bit() ){ lowValue |= (0x01<<i); } __udelay(62); } printk("lowValue is %d\n",lowValue); for(i=0; i<8; i++){ if( read_bit() ){ highValue |= (0x01<<i); } __udelay(62); } printk("highValue is %d\n",highValue); #if 0 i = highValue; i <<= 8; i = i|lowValue; value = i*0.0625; printk("kernel is %d\n",value); #endif highValue <<= 4; highValue |= ((lowValue&0xf0)>>4) ; /*拷贝内核数据到用户空间*/ copy_to_user(buffer, &highValue, sizeof(highValue)); return 0; } /*写命令,在此置空*/ static int ds18b20_write(struct file *file, const char *buffer, size_t count, loff_t * ppos) { return 0; } static int ds18b20_release(struct inode *inode,struct file *filp) { printk (KERN_INFO "device closed\n"); return 0; } static int ds18b20_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg) { return 0; } struct file_operations ds18b20_fops ={ .owner = THIS_MODULE, .open = ds18b20_open, .read = ds18b20_read, .write = ds18b20_write, .ioctl = ds18b20_ioctl, .release = ds18b20_release, }; static void ds18b20_setup_cdev(void) { int error,devno = MKDEV(ds18b20_major,0); cdev_init(&cdev,&ds18b20_fops); cdev.owner = THIS_MODULE; cdev.ops = &ds18b20_fops; error = cdev_add(&cdev,devno,1); if( error ) printk(KERN_INFO"Error %d adding ds18b20 %d\n",error,0); my_class = class_create(THIS_MODULE,"my_class"); if(IS_ERR(my_class)) { printk("Err: failed in creating class.\n"); return; } device_create(my_class,NULL,devno,NULL,"ds18b20"); } /*注册设备*/ static int ds18b20_init(void) { int result; dev = MKDEV(ds18b20_major,0); if(ds18b20_major) result = register_chrdev_region(dev,1,"ds18b20"); else{ result = alloc_chrdev_region(&dev,0,1,"ds18b20"); ds18b20_major=MAJOR(dev); } if( result < 0 ){ printk(KERN_WARNING"ds18b20:unable to get major %d\n",ds18b20_major); return result; } if(ds18b20_major == 0 ) ds18b20_major = result; ds18b20_setup_cdev(); printk("ds18b20 initialized.\n"); return 0; } static void __exit ds18b20_exit(void) { dev_t devno = MKDEV (ds18b20_major, 0); device_destroy(my_class,devno); class_destroy(my_class); cdev_del (&cdev); unregister_chrdev_region (devno, number_of_devices); printk("ds18b20_major=%d\n",ds18b20_major); printk("ds18b20 device uninstalled\n"); } module_init(ds18b20_init); module_exit(ds18b20_exit);
基于ARM-LINUX的温度传感器驱动(DS18B20) .的更多相关文章
- golang 2行代码在基于arm linux的树莓派、orangepi上运行http web服务
go语言(golang)简化了跨平台交叉编译步骤,支持在windows系统下交叉编译基于arm+linux平台的应用,运行时无需其它依赖库.以下以一个简单的http server为例,先上源码: ** ...
- 基于ARM-LINUX的温度传感器驱动-DS18B20
转载:http://blog.csdn.net/ayangke/article/details/6883244 作者:冯建,华清远见嵌入式学院讲师. DS18B20数字温度传感器接线方便,封装成后可应 ...
- 基于CC2530/CC2430 的温度采集系统--DS18B20
DS18B20是常用的温度传感器.CC2530 采集DS18B20 可以实现温度采集系统等等. 模块链接:https://item.taobao.com/item.htm?id=54130861732 ...
- 编译基于ARM LINUX的驱动模块的Makefile
KERNELDIR =/home/wenhao/platform/linux-2.6.34PWD := $(shell pwd)CROSS_COMPILE = /usr/local/arm/4.3.2 ...
- 智能开关:orange pi one(arm linux)控制继电器
大家都知道,继电器是用小电流去控制大电流运作的一种“自动开关”,在我们生活.工作中随处可见.现在的“智能家居”概念,有很多功能模块其实就是“智能开关”,远程开关.定时开关.条件触发开关等等. 下面介绍 ...
- learning docker steps(9) ----- arm linux docker 安装
参考:https://docs.docker.com/install/linux/docker-ce/ubuntu/#install-docker-ce-1 想要在arm linux上安装docker ...
- ARM Linux驱动篇 学习温度传感器ds18b20的驱动编写过程
ARM Linux驱动篇 学习温度传感器ds18b20的驱动编写过程 原文地址:http://www.cnblogs.com/NickQ/p/9026545.html 一.开发板与ds18b20的入门 ...
- 浅析基于ARM的Linux下的系统调用的实现
在Linux下系统调用是用软中断实现的,下面以一个简单的open例子简要分析一下应用层的open是如何调用到内核中的sys_open的. t8.c 1: #include <stdio.h> ...
- 【miscellaneous】【ARM-Linux开发】ARM平台基于嵌入式Linux Gstreamer 使用
1). 简介 随着ARM平台性能的日益强大和嵌入式设备的发展,对于多媒体处理如音视频播放,摄像头,流媒体处理等需求也日益增多,本文就通过几个基于嵌入式Linux下多媒体应用的示例来简单展示下使用Gst ...
- 【DSP开发】【Linux开发】基于ARM+DSP进行应用开发
针对当前应用的复杂性,SOC芯片更好能能满足应用和媒体的需求,集成众多接口,用ARM做为应用处理器进行多样化的应用开发和用户界面和接口,利用DSP进行算法加速,特别是媒体的编解码算法加速,既能够保持算 ...
随机推荐
- Oracle--SQL Developer创建连接及使用
安装好Oracle之后,有几种方式可以来管理Oracle中的数据库,首先就是登陆网页版的界面:https://localhost:1158/em,这种方式管理的东西太多,使用起来有点不方便,第二种方式 ...
- C++编译器函数模版机制剖析 - 函数模版的本质
思考:为什么函数模板能够和函数重载放在一块.C++编译器是怎样提供函数模板机制的? demo 1 #include <cstdio> #include <iostream> u ...
- [Whole Web, Nods.js, PM2] Passing environment variables to node.js using pm2
learn how to pass environment variables to your node.js app using the pm2 config file. This is usefu ...
- mvn命令安装jar包--转
有的jar通过pom配置文件不能下载,比如oracle的jdbc连接jar就是,这个时候需要通过mvn的安装命令进行手动安装jar.命令是: mvn install:install-file -Dfi ...
- VS的一部分快捷键
快捷键 功能CTRL + SHIFT + B 生成解决方案CTRL + F ...
- 使用WebView显示网页
简单的页面跳转 package com.example.webtest; import java.security.PublicKey; import android.support.v7.app.A ...
- iOS开发篇-申请开发者账号流程
1.注册一个苹果的apple id申请apple id的地址: https://appleid.apple.com/account 2.如申请公司账号,请使用以下链接免费获取邓白氏号码,以下的申请表格 ...
- mysql2csv 和 csv2mysql 工具
mysql2csv 和 csv2mysql 工具 在这里提供了两个使用 .csv 格式 的简单的 MySQL 数据库的导数据工具.csv 格式可以很容易地生成和解析,而且,也可以很容易地使用办公软件把 ...
- 在虚拟机安装64位系统提示,此主机支持Intel VT-x,但Intel VT-x处于禁用状态
进入BIOS - Security - Virtualization - Intel (R) Virtualization Technology 将 Disabled 改为 Enabled 即可
- SqlMgmt sql2008报错
标题: Microsoft SQL Server Management Studio------------------------------ 无法显示请求的对话框. --------------- ...