DS18B20实现流程

首先谈谈1-Wired总线(因为DS18B20采用此接口)
  1.首先交代常见的几种硬件通信接口
    GPIO/UART/I2C/1-Wired/SPI
    每种通信方式顺带着举例子
  2.然后谈1-Wire串行总线定义
    就七个字:一线式串行总线
    对定义进行解释
    "一线式":说明CPU和外设之间数据通信只需一根信号线
              此信号线必然是数据线
              并且数据线连接了一个上拉电阻,默认为高电平
    "串行":说明CPU和外设的数据通信一个时钟周期传输
            一个bit位,马上提出问题:
            没有时钟控制信号线,哪来的时钟呢,怎么去
            传输1个bit位呢?不像I2C总线的"低放高取",因为它有
            时钟控制信号线
            别着急待会儿后面揭晓答案
    “总线”:说明这根数据线上可以连接挂接多个外设
    此时此刻画出一个简要的硬件连接示意图,参见ds18b20.png
   3.紧接着看图提三个问题:
     1.CPU要想访问其中某个外设,那么CPU如何定位到这个外设呢?
     2.CPU如果定位到了要访问的外设,那么CPU如何通过
       一根数据线和这个外设通信呢?
     3.由于没有时钟线,如何同步双方的数据呢?
     答:答案在1-Wire总线传输协议中,协议就在外设的
         手册中,重点关注手册的时序图
         这里以DS18B20温度传感器为例,来解答以上三个
         问题,侃侃DS18B20硬件特性,只有掌握了DS18B20的硬件特性
     将来才能写出对应的驱动程序
     关键以DS18B20为例掌握一线式总线传输协议!
   4.紧接着掌控DS18B20硬件特性
     粗看:
        看连接位置
        由于X6818开发板上没有焊接一个DS18B20
     于是乎上某宝我们买一个模块单独用杜邦线连接到开发板上,
     只需三根线
     细看:先看原理图后看芯片手册
     注意用最新的底板原理图:x6818bv3.pdf
     首先看原理图,交代一下X6818,咱们自己的开发板上
     要连接一个外置的DS18B20温度传感器模块,具体连接方法如下:
     DS18B20芯片三根管脚:
     具体连接参看:连线.png,实在没把握的,先连线后拍照片发群里然后确定没问题再上电,否则烧毁!
     VDD电源:连接到X6818开发板上的5V电源
             就是电源口旁边10个插针,其中内侧的五个插针为
             5V电源接口
     GND:连接到X6818开发板上地线
         就是电源口旁边10个插针,其中外侧的五个插针为
         GND地线接口
     DQ:数据线,连接S5P6818的GPIOB10这根管脚(通过查看原理图,
        将DQ数据线连接到开发板上CPU的引脚上,并且确认引脚的编号)
        就是连接到X6818开发板GPIO接插件的最外侧的里网口最远的
        右下角的那个针脚(GPIOB10)
     并且DQ数据线必须连接一个上拉电阻,阻值必须是4.7K
     (这个电阻模块上已经焊接上,无需关注)
     注意:一线式总线接口的数据线DQ连接的CPU的引脚功能
     为GPIO功能(输入或者输出)!    
     
     最后看DS18B20芯片手册,理清它的操作过程
     目前为止应该有一个惯性思维:
     1.一开始是研究S5P6818访问DS18B20,来获取到采集的温度值
     2.通过连接图转为研究S5P6818访问数据线GPIOB10,将来S5P6818
       通过这根数据线来获取到DS18B20采集的室内温度
     3.具体CPU如何通过访问GPIOB10?关键看外设的手册的时序图
       CPU只需根据外设的时序图的要求操作GPIOB10即可,即可获取温度
       包围圈越来越小!

     首先掌握DS18B20的硬件特性:
     0.打开DS18B20芯片手册,DS18B20.pdf,掌握其相关的重要硬件信息如下:
     1.DS18B20芯片内部集成了64bit容量的ROM(只读存储器),
       存储每一个DS18B20唯一的序列码(类似身份证号)
       所以:CPU将来要想访问某个DS18B20,只需通过ROM中的
             序列码即可访问,也就是CPU只需向总线发送
             对应的DS18B20的序列码即可访问某个外设
             这也就解答了第一个问题!
             序列码类似I2C总线的外设的设备地址
    
     2.DS18B20同样芯片内部还集成了9字节容量的SRAM(不用初始化,直接访问)
       DRAM(例如DDR)这类内存必须先初始化然后才能访问
       SRAM的数据也会掉电丢失
       这个SRAM用来当成DS18B20的片内寄存器,并且9字节的
       SRAM的分配如下:
       byte0:保存采集的温度的低字节
       byte1:保存采集的温度的高字节
       结论:温度值=byte1<<8|byte0  
       结论:研究对象转移:也就是CPU只需通过一根数据线
       GPIOB10只要将DS18B20片内的SRAM中的byte0和byte1中的
       数据读取上来即可,也就获取到了对应的采集温度
       当然还有byte2....byte8,这里我们仅仅是为了
       获取温度值,所需只需关注byte0和byte1即可,
       类似CPU通过SDA和SCL两根信号线访问I2C外设片内寄存器一样!
       马上再次提出问题:CPU如何通过一根数据线GPIOB10来
       访问到byte0和byte1呢?
     3.答:要想访问byte0和byte1,芯片手册说了,必须遵循以下三步骤P10:
       1.第一步CPU向总线发送初始化复位信号,类似I2C的START信号,UART的起始位
       2.第二步CPU向总线发送ROM命令,为了找到要访问的外设,类似I2C总线发送设备地址
       3.第三步CPU向总线发送功能命令(一旦找到外设以后,下面就是读还是写还是其他功能)
       紧接着分别对着三步骤进行一一说明     
       
     4.三步骤之"CPU向总线发送初始化复位信号",类似
       I2C总线的START信号,UART的起始位
       CPU发送初始化复位信号的时序图具体参见P15,
       切记切记务必把这个时序图背下来,
       并且将来面试一定要能画出来!
       画完时序图以后,顺便写出或者说出相应的代码实现:
       void ds18b20_reset(void)
       {
        int ret;
           gpio_direction_output(PAD_GPIO_B+10, 0);//拉低
           udelay(500); //持续500us的低电平
           gpio_direction_output(PAD_GPIO_B+10, 1);//帮上拉电阻拉高
           udelay(30);//持续30us高电平
           gpio_direction_input(PAD_GPIO_B+10);//释放总线,交给DS18B20来控制
           ret = gpio_get_value(PAD_GPIO_B+10); //类似获取I2C总线的ACK信号
           if(ret==0)
              设备存在
           else
              异常
       }        
     
     5.三步骤之"CPU向总线发送ROM命令"
       如果总线上连接了多个外设,CPU只需向总线发送
       要访问的外设的序列码即可,类似I2C总线的设备地址的发送
       由于X6818连接了1个DS18B20,无需匹配,CPU直接向
       总线发送SKIP ROM命令,跳过匹配的过程,SKIP ROM的
       命令字等于0xCC,也就是CPU只需通过一根数据线将
       0xCC发送给DS18B20外设即可,问:CPU如何只需通过一根数据线将
       0xCC发送出去呢,并且要保证对方能够接收到0xCC,  
       也就是保证数据同步!并且此时此刻没有时钟控制
       信号线,没法低放高取,如何保证数据同步呢?
       注意:DS18B20数据传输从低位开始!
       
       如果不考虑数据的同步,CPU发送0xCC给外设,代码
       及其简单:
       void ds18b20_write8(unsigned char data)
       {
               //data = 0xCC(二进制11001100)
               int i;
               for (i = 0; i < 8; i++) {
                 if (data & 0x1)  //写1
                   gpio_set_value(PAD_GPIO_B+10,1);
                 else //写0
                   gpio_set_value(PAD_GPIO_B+10,0);
                 data = data >> 1;
               }
       }//由于CPU的处理速度远远快于外设,CPU执行以上
       代码的速度相当之快,外设势必无法及时处理!
       
       要想让CPU和DS18B20进行数据同步,必须严格按照
       DS18B20的操作时序图来进行,务必将涉及的四个
       时序图画出来!P16,根据时序图,得到数据同步以后的
       代码:
       void ds18b20_write8(unsigned char data)
       {
               //data = 0xCC(11001100)
               int i;
               for (i = 0; i < 8; i++) {
                 if (data & 0x1)  //写1{
                gpio_set_value(PAD_GPIO_B+8,0);
              udelay(3);
            gpio_set_value(PAD_GPIO_B+8,1);//帮上拉电阻拉高
            udelay(80);//在此期间,设备读取数据线的状态为高电平,完成写1操作
                  }else //写0 {
                           gpio_set_value(PAD_GPIO_B+8,0);
                           udelay(80);//设备读取GPIO状态,获取0,完成写0操作
                           gpio_set_value(PAD_GPIO_B+8,1);//帮上拉电阻拉高
                           udelay(3);
                       }
                 data = data >> 1;
               }
       }
       
       同理,CPU从外设DS18B20读取数据,同样也是一个bit位
       一个bit位的读,也涉及到读0和读1,根据相关的操作时序图务必会画相关数据同步的代码:
       unsigned char ds18b20_read8(void)
       {
               unsigned char data = 0;
               unsgined char ret = 0;
               int i;
               
               for (i = 0; i < 8; i++) {
                 gpio_direction_ouput(PAD_GPIO_B+8,0);//获取控制权
                 udelay(3);
                 gpio_direction_input(PAD_GPIO_B+8);//释放控制权交给外设或者上拉电阻
                 ret = gpio_get_value();//CPU获取数据线的状态,此时这个状态要不是外设给的0,要不是上拉电阻给的1
                 data |= ret << i;                       
               }
               return data;
       }
    
    6.三步骤之"CPU向总线发送功能性命令"
    DS18B20功能性命令如下:
    1.CONVET T命令,命令字等于0x44,此命令用于让DS18B20
    启动温度的采集,采集的温度,DS18B20硬件上自动保存
    温度值到SRAM的byte0和byte1中
    例如:CPU发送0x44命令给外设:ds18b20_write8(0x44);
    
    2.READ SRAM命令,命令字等于0xBE,此命令用于CPU从
    DS18B20的SRAM中读取数据
    例如:发送0xbe:ds18b20_write8(0xbe);
    
    目前知道的读取温度的思路:
    1.CPU向总线发送初始化复位信号
      ds18b20_reset();
    2.CPU向总线发送SKIP ROM命令0xCC,跳过匹配过程
        ds18b20_write8(0xCC);
    3.CPU向DS18B20先发送0x44,让他采集温度
      ds18b20_write8(0x44);
    4.CPU再向DS18B20发送0xBE,CPU读取采集的温度(位于SRAM的byte0和byte1中)
      ds18b20_write8(0xbe);
    
    问:虽然以上三步骤,都搞清楚,但是CPU读取采集的温度
    整个流程还是稀里糊涂,接下来只需看硬件操作流程图即可P13
    务必将以CPU读取温度为例,告诉考官整个操作读取温度流程:
    以CPU读取温度,QT将来要显示温度值为例,具体的流程:
    1.CPU向总线发送初始化复位信号
      ds18b20_rest();
    2.CPU向总线发送SKIP ROM命令0xCC,跳过匹配过程
      ds18b20_write8(0xCC);
    3.CPU先向总线发送温度采集命令0x44,让DS18B20进行
      硬件的温度采集,采集温度自动保存在SRAM的byte0和byte1中
      dsb18b20_write8(0x44);
    4.CPU再向总线发送初始化复位信号
      ds18b20_rest();
    5.CPU向总线发送SKIP ROM命令0xCC,跳过匹配过程
      ds18b20_write8(0xCC);
    6.CPU向总线发送读取SRAM数据命令0xBE功能性命令
      dsb18b20_write8(0xbe);
    7.CPU开始从总线上读取byte0数据
        tl = ds18b20_read8();
    8.CPU继续从总线上读取byte1数据
        th = ds18b20_read8();
    9.CPU向总线发送复位信号,停止后面SRAM数据的读取
      ds18b20_rest();
    10.最后进行温度值的最终换算:
       根据芯片手册换算出对应:
        实际温度值=(th<<8|tl)*0.0625 //给人看的
        th<<8|tl:给计算机看的,也是温度值
        
        ADC:模拟信号转数字型号转换器
        DAC:数字信号转模拟信号转换器  
    支持DS18B20硬件特性掌控完毕!
    
5.接下来就是设计软件的框架
  务必画图展示软件的框架,参见ds18b20_sw.png
  5.1.编写DS18B20硬件驱动
  ds18b20_drv.rar
  建议采用platform+GPIO库函数实现,面试时谈谈platform机制的特点,各种画图
  5.2.编写DS18B20硬件操作库和测试用例
  ds18b20_hwlib.rar
  5.3.编写QT图形界面,定时采集温度并且显示
  ehome_ds18b20.tar.bz2
  要求:利用Qtimer定时器每隔5秒读取温度,并且QT界面显示
  如果温度值高于某个阈值了,需要做出逻辑:
  启动报警器并且让LED1告警指示灯每隔
  500ms闪烁一次,并且QT界面有个按钮来回闪烁,提示用户温度过高!
  提示:QThread
  
6.最后谈谈项目实施的心得
  当我在调试DS18B20模块时,发现有时候温度读取不正常,
  但是大部分时间是没有问题,于是乎尽心跟踪调试
  拿示波器抓取数据线的时序波形,发现异常发生时,
  实际的波形不符合芯片手册的时序图,比如延时要求
  在15~60us,但是实际的波形要远远超过60us,说明
  某个任务打断了这个延时,造成延时的加长,由于
  这个模块是基于linux系统,所以想到了linux系统的
  并发和竞态问题,考虑有可能有高优先级的进程或者
  中断来打断了我的代码执行路径,于是乎利用相关的
  linux内核解决竞态的方法来解决我的bug,首先我用
  自旋锁来保护我的代码,我用自旋锁的目的就是屏蔽
  进程之间的抢占,但是呢发现还是不能解决我的问题,
  所以我的代码被打断不应该是进程的抢占,而应该
  是中断,于是乎我用另一个方法中断屏蔽,发现可以解决我的
  bug,然后我由尝试用衍生自旋锁(顺便讲讲衍生自旋锁的
  特点)也能够解决我的bug,这个仅仅是我的其中的一个调试
  心得,当然还有其他的

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
DS18B20温度传感器 * * C51 * * yajou 2008-06-28 无CRC * ********************************************************/ #include "reg51.h" #include "intrins.h" #include "DS18B20.h" /******************************************************** * us延时程序 * ********************************************************/ void Delayus(uchar us) { while(us--); //12M,一次6us,加进入退出14us(8M晶振,一次9us) } /******************************************************** * DS18B20初始化 * ********************************************************/ bit Ds18b20_Init(void) //存在返0,否则返1 { bit temp = 1; uchar outtime = ReDetectTime; //超时时间 while(outtime-- && temp) { Delayus(10); //(250)1514us时间可以减小吗 ReleaseDQ(); Delay2us(); PullDownDQ(); Delayus(100); //614us(480-960) ReleaseDQ(); Delayus(10); //73us(>60) temp = dq; Delayus(70); //us } return temp; } /******************************************************** * 写bit2DS18B20 * ********************************************************/ void Ds18b20_WriteBit(bit bitdata) { if(bitdata) { PullDownDQ(); Delay2us(); //2us(>1us) ReleaseDQ(); //(上述1-15) Delayus(12); //86us(45- x,总时间>60) }else { PullDownDQ(); Delayus(12); //86us(60-120) } ReleaseDQ(); Delay2us(); //2us(>1us) } /******************************************************** * 写Byte DS18B20 * ********************************************************/ void Ds18b20_WriteByte(uchar chrdata) { uchar ii; for(ii = 0; ii < 8; ii++) { Ds18b20_WriteBit(chrdata & 0x01); chrdata >>= 1; } } /******************************************************** * 写 DS18B20 * ********************************************************/ //void Ds18b20_Write(uchar *p_readdata, uchar bytes) //{ // while(bytes--) // { // Ds18b20_WriteByte(*p_readdata); // p_readdata++; // } //} /******************************************************** * 读bit From DS18B20 * ********************************************************/ bit Ds18b20_ReadBit(void) { bit bitdata; PullDownDQ(); Delay2us(); //2us( >1us) ReleaseDQ(); Delay8us(); //8us( <15us) bitdata = dq; Delayus(7); //86us(上述总时间要>60us) return bitdata; } /******************************************************** * 读Byte DS18B20 * ********************************************************/ uchar Ds18b20_ReadByte(void) { uchar ii,chardata; for(ii = 0; ii < 8; ii++) { chardata >>= 1; if(Ds18b20_ReadBit()) chardata |= 0x80; } return chardata; } /******************************************************** * 读 DS18B20 ROM * ********************************************************/ bit Ds18b20_ReadRom(uchar *p_readdata) //成功返0,失败返1 { uchar ii = 8; if(Ds18b20_Init()) return 1; Ds18b20_WriteByte(ReadROM); while(ii--) { *p_readdata = Ds18b20_ReadByte(); p_readdata++; } return 0; } /******************************************************** * 读 DS18B20 EE * ********************************************************/ bit Ds18b20_ReadEE(uchar *p_readdata) //成功返0,失败返1 { uchar ii = 2; if(Ds18b20_Init()) return 1; Ds18b20_WriteByte(SkipROM); Ds18b20_WriteByte(ReadScr); while(ii--) { *p_readdata = Ds18b20_ReadByte(); p_readdata++; } return 0; } /******************************************************** * 温度采集计算 * ********************************************************/ bit TempCal(float *p_wendu) //成功返0,失败返1 (温度范围-55 --- +128) { uchar temp[9],ii; uint tmp; float tmpwendu; TR1 = 0; TR0 = 0; //读暂存器和CRC值----------------------- if(Ds18b20_ReadEE(temp)) { TR1 = 1; TR0 = 1; return 1; } //------------------------------------- //CRC校验------------------------------ // //此处应加入CRC校验等 // // //------------------------------------- //使温度值写入相应的wendu[i]数组中----- for(ii = i; ii > 0; ii--) { p_wendu++; } i++; if(i > 4) i = 0; //------------------------------------- //温度正负数处理----------------------- // //------------------------------------- //温度计算----------------------------- tmp = temp[1]; // tmp <<= 8; // tmp |= temp[0]; //组成温度的两字节合并 tmpwendu = tmp; *p_wendu = tmpwendu / 16; //------------------------------------- //开始温度转换------------------------- if(Ds18b20_Init()) { TR1 = 1; TR0 = 1; return 1; } Ds18b20_WriteByte(SkipROM); Ds18b20_WriteByte(Convert); ReleaseDQ(); //寄生电源时要拉高DQ //------------------------------------ TR1 = 1; TR0 = 1; return 0; } //////////DS18B20.h///////////////////////// /******************************************************** * I/O口定义 * ********************************************************/ sbit dq = P1^3; sbit dv = P1^4; //DS18B20强上拉电源 /******************************************************** * 命令字定义 * ********************************************************/ #define uchar unsigned char #define uint unsigned int #define ReleaseDQ() dq = 1; //上拉/释放总线 #define PullDownDQ() dq = 0; //下拉总线 #define Delay2us() _nop_();_nop_(); //延时2us,每nop 1us #define Delay8us() _nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_(); //设置重复检测次次数,超出次数则超时 #define ReDetectTime 20 //ds18b20命令 #define SkipROM 0xCC #define MatchROM 0x55 #define ReadROM 0x33 #define SearchROM 0xF0 #define AlarmSearch 0xEC #define Convert 0x44 #define WriteScr 0x4E #define ReadScr 0xBE #define CopyScr 0x48 #define RecallEE 0xB8 #define ReadPower 0xB4 /******************************************************** * 函数 * ********************************************************/ void Delayus(uchar us); //void Dog(void); bit Ds18b20_Init(void); //DS18B20初始化,存在返0,否则返1 void Ds18b20_WriteBit(bit bitdata); //写bit2DS18B20 void Ds18b20_WriteByte(uchar chrdata); //写Byte DS18B20 void Ds18b20_Write(uchar *p_readdata, uchar bytes); //写 DS18B20 bit Ds18b20_ReadBit(void); //读bit From DS18B20 uchar Ds18b20_ReadByte(void); //读Byte DS18B20 bit Ds18b20_ReadRom(uchar *p_readdata); //读 DS18B20 ROM:成功返0,失败返1 bit Ds18b20_ReadEE(uchar *p_readdata); //读 DS18B20 EE :成功返0,失败返1 bit TempCal(float *p_wendu); //成功返0,失败返1 (温度范围-55 --- +128) [目录] 第一章 前言 第二章 设计方案 第三章 数字温度传感器芯片特性 第四章 AT89S52单片机简介 第五章 单片机驱动蜂鸣器原理 第六章 单片机驱动继电器原理 第七章 按键设计 第八章 数码管显示电路 附录 1.源程序 2.电路图 [摘要] 应用数字温度传感器DS18B20设计的智能温度控制系统,实现方便、精度高、功耗低、微型化、抗干扰能力强,可根据不同需要用于各种温度监控及其他各种温度测控系统中。简单的外围电路主要依靠单片机的程序控制,实现温度的实时采集与比较,温度值的十进制数转换,-55°C ~125°C实时的温度显示及上下限温度值显示,键盘对上下限温度的设定,各种数据处理及报警温度的判断,单片机对继电器的驱动实现相应的加热、制冷控制。 在单片机程序的控制下,新一代的可编程数字温度传感器DS18B20完成其温度的转化和相应的数据处理与比较;选择简单的独立式按键,简化程序。大量应用PNP三极管的开关作用和电流的放大作用,实现单片机I/O口小电流的TTL电平对外围器件的控制。加热、制冷电机启动指示灯及各种保护,恒温指示灯,和各种报警声构成人性化智能温控系统。 [正文] 第一章 前言 本论文介绍单片机结合DS18B20设计的智能温度控制系统,系统用一种新型的“一总线”可编程数字温度传感器(DS18B20),不需复杂的信号调理电路和A/D转换电路能直接与单片机完成数据采集和处理,实现方便、精度高、功耗低、微型化、抗干扰能力强,可根据不同需要用于各种温度监控及其他各种温度测控系统中。 美国DALLAS最新单线数字温度传感器DS18B20,具有微型化低功耗、高性能、可组网等优点,新的“一线器件”体积更小、适用电压更宽、更经济 Dallas 半导体公司的数字化温度传感器DS1820是世界上第一片支持 “一线总线”接口的温度传感器。一线总线独特而且经济的特点,使用户可轻松地组建传感器网络,为测量系统的构建引入全新概念。DS18B20的测温分辨率较高,DS18B20可直接将温度转化成串行数字信号,因此特别适合和单片机配合使用,直接读取温度数据。目前DS18B20数字温度传感器已经广泛应用于恒温室、粮库、计算机机房。测量温度范围为 -55°C~+125°C,在-10~+85°C范围内,误差为±0.5°C。现场温度直接以“一线总线”的数字方式传输,大大提高了系统的抗干扰性。适合于恶劣环境的现场温度测量,如:环境控制、设备或过程控制、测温类消费电子产品等。新的产品支持3V~5.5V的电压范围,使系统设计更灵活、方便。而且新一代产品更便宜,体积更小。 DS18B20可以程序设定9~12位的分辨率,精度为0.0625°C。可选更小的封装方式,更宽的电压适用范围。分辨率设定,及用户设定的报警温度存储在EEPROM中,掉电后依然保存。DS18B20的性能是新一代产品中最好的!性能价格比也非常出色!DS18B20使电压、特性及封装有更多的选择,让我们可以构建适合自己的经济的测温系统。 在传统的模拟信号远距离温度测量系统中,需要很好的解决引线误差补偿问题、多点测量切换误差问题和放大电路零点漂移误差问题等技术问题,才能够达到较高的测量精度。另外一般监控现场的电磁环境都非常恶劣,各种干扰信号较强,模拟温度信号容易受到干扰而产生测量误差,影响测量精度。因此,在温度测量系统中,采用抗干扰能力强的新型数字温度传感器是解决这些问题的最有效方案,新型数字温度传感器DS18B20具有体积更小、精度更高、适用电压更宽、采用一线总线、可组网等优点,在实际应用中取得了良好的测温效果。传统的测温元件测出的一般都是电压,再转换成对应的温度,需要比较多的外部硬件支持,电路复杂,软件调试复杂,制作成本高。所以本人改用一种智能传感器DS18B20作为检测元件,可以直接读出被测温度值。1线制与单片机相连,减少了外部硬件电路,具有低成本和易使用的特点。 [参考文献] [1] 童诗白、华成英.模拟电子技术基础.高等教育出版社,2000 [2] 阉石.数字电子技术基础.高等教育出版社,1998 [3] 李朝青.单片机原理与接口技术.北京航空航天大学出版社,2000 [4] 楼然苗、李光飞.单片机课程设计指导.电子工业出版社,2007 [5] Intel. MCS-51 Family of Single Chip Microcomputers User’s Manual.1990 [6] Keil Software Company. Cx51 Compiler User’s Guide. 2001 [7] 李群芳.单片机微型计算机与接口技术.电子工业出版社,1997 [8] 全国大学生电子设计竞赛——1994年获奖作品选编 [9] 肖忠祥.数据采集原理.西北工业大学出版社,2001 [10] ATMEL公司 AT89S52的技术手册 [11] 吴金戌、沈庆阳、郭庭吉.单片机实践与应用.北京:清华大学出版社 [12] 王为青、邱文勋.51单片机应用开发案例精选.人民邮电出版社,2007  TS-18B20 数字温度传感器(www.ftco01.cn)   本公司最新推出TS-18B20数字温度传感器,该产品采用美国DALLAS公司生产的 DS18B20可组网数字温度传感器芯片封装而成,具有耐磨耐碰,体积小,使用方便,封装形式多样,适用于各种狭小空间设备数字测温和控制领域。   1: 技术性能描述   1.1 独特的单线接口方式,DS18B20在与微处理器连接时仅需要一条口线即可实现微处理器与DS18B20的双向通讯。   1.2 测温范围 -55℃~+125℃,固有测温分辨率0.5℃。   1.3 支持多点组网功能,多个DS18B20可以并联在唯一的三线上,实现多点测温   1.4 工作电源: 3~5V/DC   1.5 在使用中不需要任何外围元件   1.6 测量结果以9~12位数字量方式串行传送   1.7 不锈钢保护管直径 Φ6   1.8 适用于DN15~25, DN40~DN250各种介质工业管道和狭小空间设备测温   1.9 标准安装螺纹 M10X1, M12X1.5, G1/2”任选   1.10 PVC电缆直接出线或德式球型接线盒出线,便于与其它电器设备连接。   2:应用范围   2.1 该产品适用于冷冻库,粮仓,储罐,电讯机房,电力机房,电缆线槽等测温和控制领域   2.2 轴瓦,缸体,纺机,空调,等狭小空间工业设备测温和控制。   2.3 汽车空调、冰箱、冷柜、以及中低温干燥箱等。   2.5 供热/制冷管道热量计量,中央空调分户热能计量和工业领域测温和控制   3:产品型号与规格   型 号 测温范围 安装螺纹 电缆长度 适用管道   TS-18B20 -55~125 无 1.5 m   TS-18B20A -55~125 M10X1 1.5m DN15~25   TS-18B20B -55~125 1/2”G 接线盒 DN40~ 60   4:接线说明   特点 独特的一线接口,只需要一条口线通信 多点能力,简化了分布式温度传感应用 无需外部元件 可用数据总线供电,电压范围为3.0 V至5.5 V 无需备用电源 测量温度范围为-55 ° C至+125 ℃ 。华氏相当于是-67 ° F到257华氏度 -10 ° C至+85 ° C范围内精度为±0.5 ° C   温度传感器可编程的分辨率为9~12位 温度转换为12位数字格式最大值为750毫秒 用户可定义的非易失性温度报警设置 应用范围包括恒温控制,工业系统,消费电子产品温度计,或任何热敏感系统   描述该DS18B20的数字温度计提供9至12位(可编程设备温度读数。信息被发送到/从DS18B20 通过1线接口,所以中央微处理器与DS18B20只有一个一条口线连接。为读写以及温度转换可以从数据线本身获得能量,不需要外接电源。 因为每一个DS18B20的包含一个独特的序号,多个ds18b20s可以同时存在于一条总线。这使得温度传感器放置在许多不同的地方。它的用途很多,包括空调环境控制,感测建筑物内温设备或机器,并进行过程监测和控制。   8引脚封装 TO-92封装 用途 描述   5 1 接地 接地   4 2 数字 信号输入输出,一线输出:源极开路   3 3 电源 可选电源管脚。见"寄生功率"一节细节方面。电源必须接地,为行动中,寄生虫功率模式。   不在本表中所有管脚不须接线 。   概况框图图1显示的主要组成部分DS18B20的。DS18B20内部结构主要由四部分组成:64位光刻ROM、温度传感器、非挥发的温度报警触发器TH和TL、配置寄存器。该装置信号线高的时候,内部电容器 储存能量通由1线通信线路给片子供电,而且在低电平期间为片子供电直至下一个高电平的到来重新充电。 DS18B20的电源也可以从外部3V-5 .5V的电压得到。   DS18B20采用一线通信接口。因为一线通信接口,必须在先完成ROM设定,否则记忆和控制功能将无法使用。主要首先提供以下功能命令之一: 1 )读ROM, 2 )ROM匹配, 3 )搜索ROM, 4 )跳过ROM, 5 )报警检查。这些指令操作作用在没有一个器件的64位光刻ROM序列号,可以在挂在一线上多个器件选定某一个器件,同时,总线也可以知道总线上挂有有多少,什么样的设备。   若指令成功地使DS18B20完成温度测量,数据存储在DS18B20的存储器。一个控制功能指挥指示DS18B20的演出测温。测量结果将被放置在DS18B20内存中,并可以让阅读发出记忆功能的指挥,阅读内容的片上存储器。温度报警触发器TH和TL都有一字节EEPROM 的数据。如果DS18B20不使用报警检查指令,这些寄存器可作为一般的用户记忆用途。在片上还载有配置字节以理想的解决温度数字转换。写TH,TL指令以及配置字节利用一个记忆功能的指令完成。通过缓存器读寄存器。所有的数据都读,写都是从最低位开始。   DS18B20有4个主要的数据部件:   (1)光刻ROM中的64位序列号是出厂前被光刻好的,它可以看作是该DS18B20的地址序列码。64位光刻ROM的排列是:开始8位(28H)是产品类型标号,接着的48位是该DS18B20自身的序列号,最后8位是前面56位的循环冗余校验码(CRC=X8+X5+X4+1)。光刻ROM的作用是使每一个DS18B20都各不相同,这样就可以实现一根总线上挂接多个DS18B20的目的。   (2) DS18B20中的温度传感器可完成对温度的测量,以12位转化为例:用16位符号扩展的二进制补码读数形式提供,以0.0625℃/LSB形式表达,其中S为符号位。   表1 DS18B20温度值格式表   4.3.1   DS18B20的管脚排列如图4.4所示。   图4.4DS18B20的管脚排列如图   DS18B20内部结构主要由四部分组成:64位光刻ROM,温度传感器,温度报警触发器TH和TL,配置寄存器。DS18B20内部结构图如图4.5所示。   图4.5 DS18B20内部结构图   4.3.2存储器   DS18B20的存储器包括高速暂存器RAM和可电擦除RAM,可电擦除RAM又包括温度触发器TH和TL,以及一个配置寄存器。存储器能完整的确定一线端口的通讯,数字开始用写寄存器的命令写进寄存器,接着也可以用读寄存器的命令来确认这些数字。当确认以后就可以用复制寄存器的命令来将这些数字转移到可电擦除RAM中。当修改过寄存器中的数时,这个过程能确保数字的完整性。   高速暂存器RAM是由8个字节的存储器组成;第一和第二个字节是温度的显示位。第三和第四个字节是复制TH和TL,同时第三和第四个字节的数字可以更新;第五个字节是复制配置寄存器,同时第五个字节的数字可以更新;六、七、八三个字节是计算机自身使用。用读寄存器的命令能读出第九个字节,这个字节是对前面的八个字节进行校验。存储器的结构图如图4.6所示。   图4.6 存储器的结构图   4.3.3 64-位光刻ROM   64位光刻ROM的前8位是DS18B20的自身代码,接下来的48位为连续的数字代码,最后的8位是对前56位的CRC校验。64-位的光刻ROM又包括5个ROM的功能命令:读ROM,匹配ROM,跳跃ROM,查找ROM和报警查找。64-位光刻ROM的结构图如图4.7所示。   图4.7位64-位光刻ROM的结构图   4.3.4 DS18B20外部电源的连接方式   DS18B20可以使用外部电源VDD,也可以使用内部的寄生电源。当VDD端口接3.0V—5.5V的电压时是使用外部电源;当VDD端口接地时使用了内部的寄生电源。无论是内部寄生电源还是外部供电,I/O口线要接5KΩ左右的上拉电阻。 连接图如图4.8、图4.9所示。   图4.8 使用寄生电源的连接图   图4.9外接电源的连接图   4.3.4 DS18B20温度处理过程   4.3.4.1配置寄存器   配置寄存器是配置不同的位数来确定温度和数字的转化。配置寄存器的结构图如图4.10所示。   图4.10 配置寄存器的结构图   由图4.9可以知道R1,R0是温度的决定位,由R1,R0的不同组合可以配置为9位,10位,11位,12位的温度显示。这样就可以知道不同的温度转化位所对应的转化时间,四种配置的分辨率分别为0.5℃,0.25℃,0.125℃和0.0625℃,出厂时以配置为12位。温度的决定配置图如图8所示。   图4.11 温度的决定配置图   4.3.4.2 温度的读取   DS18B20在出厂时以配置为12位,读取温度时共读取16位,所以把后11位的2进制转化为10进制后在乘以0.0625便为所测的温度,还需要判断正负。前5个数字为符号位,当前5位为1时,读取的温度为负数;当前5位为0时,读取的温度为正数。16位数字摆放是从低位到高位,温度的关系图如图4.12所示。   图4.12为温度的关系图   4.3.4.3.DS18B20控制方法   DS18B20有六条控制命令,如表4.1所示:   表4.1 为DS18B20有六条控制命令   指 令 约定代码 操 作 说 明   温度转换 44H 启动DS18B20进行温度转换   读暂存器 BEH 读暂存器9个字节内容   写暂存器 4EH 将数据写入暂存器的TH、TL字节   复制暂存器 48H 把暂存器的TH、TL字节写到E2RAM中   重新调E2RAM B8H 把E2RAM中的TH、TL字节写到暂存器TH、TL字节   读电源供电方式 B4H 启动DS18B20发送电源供电方式的信号给主CPU   4.3.4.4 DS18B20的初始化   (1) 先将数据线置高电平“1”。   (2) 延时(该时间要求的不是很严格,但是尽可能的短一点)   (3) 数据线拉到低电平“0”。   (4) 延时750微秒(该时间的时间范围可以从480到960微秒)。   (5) 数据线拉到高电平“1”。   (6) 延时等待(如果初始化成功则在15到60毫秒时间之内产生一个由DS18B20所返回的低电平“0”。据该状态可以来确定它的存在,但是应注意不能无限的进行等待,不然会使程序进入死循环,所以要进行超时控制)。   (7) 若CPU读到了数据线上的低电平“0”后,还要做延时,其延时的时间从发出的高电平算起(第(5)步的时间算起)最少要480微秒。   (8) 将数据线再次拉高到高电平“1”后结束。   其时序如图4.13所示:   图4.13 初始化时序图   4.3.4.5 DS18B20的写操作   (1) 数据线先置低电平“0”。   (2) 延时确定的时间为15微秒。   (3) 按从低位到高位的顺序发送字节(一次只发送一位)。   (4) 延时时间为45微秒。   (5) 将数据线拉到高电平。   (6) 重复上(1)到(6)的操作直到所有的字节全部发送完为止。   (7) 最后将数据线拉高。   DS18B20的写操作时序图如图4.14所示。   图4.14 DS18B20的写操作时序图   4.3.4.6 DS18B20的读操作   (1)将数据线拉高“1”。   (2)延时2微秒。   (3)将数据线拉低“0”。   (4)延时15微秒。   (5)将数据线拉高“1”。   (6)延时15微秒。   (7)读数据线的状态得到1个状态位,并进行数据处理。   (8)延时30微秒。   DS18B20的读操作时序图如图4.15所示。   图1.15 DS18B20的读操作图
DS18B20是一种数字温度传感器,使用时需要连接到控制器的GPIO口,然后通过读取DS18B20的数据来获取温度值。以下是DS18B20的代码流程: 1. 初始化GPIO口,将其设置为输出模式,输出高电平。 2. 延时500毫秒。 3. 将GPIO口设置为输入模式,读取DS18B20的响应信号。 4. 若DS18B20响应信号为低电平,则继续执行;否则等待一段时间再重试。 5. 发送读取温度命令,读取DS18B20传输的数据。 6. 将读取到的数据转换为温度值。 7. 返回温度值。 下面是一个基于STM32的DS18B20代码示例: ``` #include "ds18b20.h" #include "delay.h" #include "gpio.h" #define DS18B20_PIN GPIO_PIN_0 #define DS18B20_PORT GPIOA #define DS18B20_SKIP_ROM 0xCC #define DS18B20_CONVERT_T 0x44 #define DS18B20_READ_SCRATCHPAD 0xBE void ds18b20_init(void) { // 初始化GPIO口 gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_OUTPUT_PP, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_SET); } uint8_t ds18b20_reset(void) { // 将GPIO口设置为输出模式,输出低电平 gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_OUTPUT_PP, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_RESET); delay_us(480); // 将GPIO口设置为输入模式,等待DS18B20响应信号 gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_INPUT, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); delay_us(80); // 若DS18B20响应信号为低电平,则继续执行;否则等待一段时间再重试 if (gpio_read_pin(DS18B20_PORT, DS18B20_PIN) == GPIO_PIN_RESET) { delay_us(400); return 1; } else { return 0; } } void ds18b20_write_byte(uint8_t data) { for (int i = 0; i < 8; i++) { gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_OUTPUT_PP, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_RESET); delay_us(2); if (data & (1 << i)) { gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_OUTPUT_PP, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_SET); } else { gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_OUTPUT_PP, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_RESET); } delay_us(60); gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_INPUT, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_SET); delay_us(2); } } uint8_t ds18b20_read_byte(void) { uint8_t data = 0; for (int i = 0; i < 8; i++) { gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_OUTPUT_PP, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_RESET); delay_us(2); gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_INPUT, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); data |= gpio_read_pin(DS18B20_PORT, DS18B20_PIN) << i; delay_us(60); gpio_init(DS18B20_PORT, DS18B20_PIN, GPIO_MODE_INPUT, GPIO_NOPULL, GPIO_SPEED_FREQ_LOW); gpio_write_pin(DS18B20_PORT, DS18B20_PIN, GPIO_PIN_SET); delay_us(2); } return data; } float ds18b20_read_temp(void) { float temp = 0; if (ds18b20_reset()) { ds18b20_write_byte(DS18B20_SKIP_ROM); ds18b20_write_byte(DS18B20_CONVERT_T); delay_ms(800); ds18b20_reset(); ds18b20_write_byte(DS18B20_SKIP_ROM); ds18b20_write_byte(DS18B20_READ_SCRATCHPAD); uint8_t low_byte = ds18b20_read_byte(); uint8_t high_byte = ds18b20_read_byte(); int16_t raw_temp = (high_byte << 8) | low_byte; temp = (float)raw_temp / 16.0; } return temp; } ``` 以上代码仅供参考,具体实现还需要根据不同的硬件平台进行相应的修改和优化。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值