第一部分:DS18B20的封装和管脚定义
首先,我们来认识一下DS18B20这款芯片的外观和针脚定义,DS18B20芯片的常见封装为TO-92,也就是普通直插三极管的样子,当然也可以找到以SO(DS18B20Z)和μSOP(DS18B20U)形式封装的产品,下面为DS18B20各种封装的图示及引脚图。
了解了这些该芯片的封装形式,下面就要说到各个管脚的定义了,如下表即为该芯片的管脚定义:
上面的表中提到了一个“奇怪”的词——“寄生电源”,那我有必要说明一下了,DS18B20芯片可以工作在“寄生电源模式”下,该模式允许DS18B20工作在无外部电源状态,当总线为高电平时,寄生电源由单总线通过VDD引脚,此时DS18B20可以从总线“窃取”能量,并将“偷来”的能量储存到寄生电源储能电容(Cpp)中,当总线为低电平时释放能量供给器件工作使用。所以,当DS18B20工作在寄生电源模式时,VDD引脚必须接地。
第二部分:DS18B20的多种电路连接方式
如下面的两张图片所示,分别为外部供电模式下单只和多只DS18B20测温系统的典型电路连接图。
(1)外部供电模式下的单只DS18B20芯片的连接图
(2)外部供电模式下的多只DS18B20芯片的连接图
这里需要说明的是,DS18B20芯片通过达拉斯公司的单总线协议依靠一个单线端口通讯,当全部器件经由一个三态端口或者漏极开路端口与总线连接时,控制线需要连接一个弱上拉电阻。在多只DS18B20连接时,每个DS18B20都拥有一个全球唯一的64位序列号,在这个总线系统中,微处理器依靠每个器件独有的64位片序列号辨认总线上的器件和记录总线上的器件地址,从而允许多只DS18B20同时连接在一条单线总线上,因此,可以很轻松地利用一个微处理器去控制很多分布在不同区域的DS18B20,这一特性在环境控制、探测建筑物、仪器等温度以及过程监测和控制等方面都非常有用。
对于DS18B20的电路连接,除了上面所说的传统的外部电源供电时的电路连接图,DS18B20也可以工作在“寄生电源模式”,而下图则表示了DS18B20工作在“寄生电源模式”下的电路连接图。没错,这样就可以使DS18B20工作在寄生电源模式下了,不用额外的电源就可以实时采集到位于多个地点的温度信息了。
第三部分:DS18B20内部寄存器解析及工作原理
介绍完DS18B20的封装、针脚定义和连接方式后,我们有必要了解DS18B20芯片的各个控制器、存储器的相关知识,如下图所示,为DS18B20内部主要寄存器的结果框图。
结合图中的内部寄存器框图,我们先简单说一下DS18B20芯片的主要寄存器工作流程,而在对DS18B20工作原理进行详细说明前,有必要先上几张相关图片:
(1)DS18B20内部寄存器结构图
(2)DS18B20主要寄存器数据格式图示
(3)DS18B20通讯指令图
了解了这些内部结构和细节,下面说一下DS18B20芯片的工作原理。
DS18B20启动后将进入低功耗等待状态,当需要执行温度测量和AD转换时,总线控制器(多为单片机)发出[44H]指令完成温度测量和AD转换(其他功能指令见上面的指令表),DS18B20将产生的温度数据以两个字节的形式存储到高速暂存器的温度寄存器中,然后,DS18B20继续保持等待状态。当DS18B20芯片由外部电源供电时,总线控制器在温度转换指令之后发起“读时隙”(详见本帖的“DS18B20时隙图”),从而读出测量到的温度数据通过总线完成与单片机的数据通讯(DS18B20正在温度转换中由DQ引脚返回0,转换结束则返回1。如果DS18B20由寄生电源供电,除非在进入温度转换时总线被一个强上拉拉高,否则将不会有返回值)。另外,DS18B20在完成一次温度转换后,会将温度值与存储在TH(高温触发器)和TL(低温触发器)中各一个字节的用户自定义的报警预置值进行比较,寄存器中的S标志位(详见寄存器格式图示中的“TH和TL寄存器格式”图示)指出温度值的正负(S=0时为正,S=1时为负),如果测得的温度高于TH或者低于TL数值,报警条件成立,DS18B20内部将对一个报警标识置位,此时,总线控制器通过发出报警搜索命令[ECH]检测总线上所有的DS18B20报警标识,然后,对报警标识置位的DS18B20将响应这条搜索命令。
第四部分:针对DS18B20的单片机编程
针对DS18B20的编程,可以理解为总线控制器通过相关指令操作器件或者器件中的相应寄存器,从而完成器件也总线控制器的数据通信,所以要真正搞定DS18B20的通讯编程,还需要详细的了解该芯片的各种寄存器结构、寄存器数据格式和相关的指令系统,下面我们就结合上面图示,说说DS18B20的内部存储器结构。
DS18B20的每个暂存器都有8bit存储空间,用来存储相应数据,其中byte0和byte1分别为温度数据的低位和高位,用来储存测量到的温度值,且这两个字节都是只读的;byte2和byte3为TH、TL告警触发值的拷贝,可以在从片内的电可擦可编程只读存储器EEPROM中读出,也可以通过总线控制器发出的[48H]指令将暂存器中TH、TL的值写入到EEPROM,掉电后EEPROM中的数据不会丢失;byte4的配置寄存器用来配置温度转换的精确度(最大为12位精度);byte5、6、7为保留位,禁止写入;byte8亦为只读存储器,用来存储以上8字节的CRC校验码。
参考上面的DS18B20通讯指令图,即为DS18B20芯片中主要寄存器的数据格式和必要的个别标识位说明,只要做到对寄存器数据精准的控制,就可以很容易的完成DS18B20的程序编写,而对于总线控制器发出的控制指令,我们需要知道,DS18B20的指令包括ROM指令和功能指令,其中ROM指令用来进行ROM的操作,而功能指令则可以控制DS18B20完成温度转换,寄存器操作等功能性工作。一旦总线控制器检测到一个存在脉冲,它就会发出一条ROM指令,如果总线上挂载多只DS18B20,这些指令将利用器件独有的64位ROM片序列码选出特定的要进行操作的器件,同样,这些指令也可以识别哪些器件符合报警条件等。在总线控制器发给要连接的DS18B20一条ROM指令后,就可以发送一条功能指令完成相关的工作了,也就是说,总线控制器在发起一条DS18B20功能指令前,需要首先发出一条ROM指令。了解了这些功能指令的功能和用法,再对DS18B20编程就容易多了!~
第五部分:DS18B20芯片的两点使用心得
(1)对TH(高温触发寄存器)和TL(低温触发寄存器)的操作心得
针对于DS18B20中TH(高温触发寄存器)和TL(低温触发寄存器),可以找到的代码资料很少,而如果在某一测温系统中需要用到TH和TL寄存器时,其实不必觉得无从下手,参见本帖中的“DS18B20寄存器结构”,总线控制器的读操作将从位0开始逐步向下读取数据,直到读完位8,而且TH和TL寄存器的内部结构和数据格式和片内其他寄存器是相同的,当然,针对TH和TL寄存器的读写和其他片内寄存器的读写也是相同的,所以在实际应用中,当DS18B20初始化完成后,首先通过总线控制器发出的[B8H]指令将EEPROM中保存的数据召回到暂存器的TH和TL中,然后通过总线控制器发出的“读时隙”对器件暂存器进行读操作,只要将读到的每8bit数据及时获取,就可以很容易地通过总线控制器读出TH和TL寄存器数据;总线控制器对器件的写操作原理亦然,换句话说,只要掌握了其他寄存器的操作编程,就完全可以很容易地对TH和TL这两个报警值寄存器进行读写操作。同时,可以通过[48H]指令将TH和TL寄存器数据拷贝到EEPROM中进行保存。
(2)对DS18B20通讯时隙的掌握心得
在由DS18B20芯片构建的温度检测系统中,采用达拉斯公司独特的单总线数据通讯方式,允许在一条总线上挂载多个DS18B20,那么,在对DS18B20的操作和控制中,由总线控制器发出的时隙信号就显得尤为重要。如下图所示,分别为DS18B20芯片的上电初始化时隙、总线控制器从DS18B20读取数据时隙、总线控制器向DS18B20写入数据时隙的示意图,在系统编程时,一定要严格参照时隙图中的时间数据,做到精确的把握总线电平随时间(微秒级)的变化,才能够顺利地控制和操作DS18B20。另外,需要注意到不同单片机的机器周期是不尽相同的,所以,程序中的延时函数并不是完全一样,要根据单片机不同的机器周期有所改动。在平常的DS18B20程序调试中,若发现诸如温度显示错误等故障,基本上都是由于时隙的误差较大甚至时隙错误导致的,在对DS18B20编程时需要格外注意。
上电初始化时隙图
数据读取时通讯总线的时隙图
数据写入时通讯总线的时隙图
DS18B20测温芯片做的数码管显示温度计程序
下列程序也是用网上的程序进行修改,主要改动部分是显示输出部分,而温度转换是采用的查表法(具体原理还没有看懂),但该程序经过编译,100%通过。STC12C5A60S2.h头文件下载:http://www.51hei.com/mcu/2564.html
#include < STC12C5A60S2.h >
#include < intrins.h >
#define uchar unsigned char
#define uint unsigned int
bit presence ;
//***************************************************************************************************//
sbit DQ = P1^6 ; //定义DS18B20端口DQ
sbit LED_A =P4 ^ 4; //设置LED点阵屏连接的I/O口
sbit LED_B =P0 ^ 6; //设置LED点阵屏连接的I/O口
sbit LED_C =P3 ^ 0; //设置LED点阵屏连接的I/O口
sbit LED_D =P3 ^ 2; //设置LED点阵屏连接的I/O口
sbit LED_E =P3 ^ 3; //设置LED点阵屏连接的I/O口
sbit LED_F =P4 ^ 5; //设置LED点阵屏连接的I/O口
sbit LED_G =P4 ^ 7; //设置LED点阵屏连接的I/O口
sbit LED_DP =P3 ^ 1; //设置LED点阵屏连接的I/O口
sbit LED1= P2^7 ; //定义LED数码管位脚
sbit LED2= P4^6 ;
sbit LED3= P0^7 ;
sbit LED4= P1^7 ;
//***************************************************************************************************//
unsigned char data temp_data[2] = {0x00,0x00} ;
unsigned char data display[5] = {0x00,0x00,0x00,0x00,0x00} ;
unsigned char code ditab[16] = {0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,
0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09} ;
unsigned char code mytab[4] = {0xF9,0xB0,0x92,0x99} ;//纯粹用来测试,可删除
//***************************************************************************************************//
const uchar tab[]={ /* 根据共阴极字型编码表获取0~9,A~B字型代码 */
0x3f,0x06,0x5b,0x4f,//0~3
0x66,0x6D,0x7D,0x07,//4~7
0x7F,0x6F,0x77,0x7C,//8~b
0x39,0x5E,0x79,0x71,//c~f
0x00//mie
};
//***************************************************************************************************//
const uchar tab1[]={ /* 根据共阳极字型编码表获取0~9,A~B字型代码 */
0xC0,0xF9,0xA4,0xB0,//0~3
0x99,0x92,0x82,0xF8,//4~7
0x80,0x90,0x88,0x83,//8~b
0xC6,0xA1,0x86,0x8E,//c~f
0xff//mie
};
#define delayNOP() ; {_nop_() ;_nop_();_nop_() ;_nop_() ;} ;
/*******************************************************************/
void delay1(int ms)
{
unsigned char y ;
while(ms--)
{
for(y = 0 ; y<250 ; y++)
{
_nop_() ;
_nop_() ;
_nop_() ;
_nop_() ;
}
}
}
/******************************************************************/
/*us级延时函数 */
/*******************************************************************/
void Delay(unsigned int num) //延时6us,误差 0us
{
unsigned char a;
while(num--!=0)
{for(a=15;a>0;a--);}
}
/*void Delay(unsigned int num)
{
while( --num ) ;
}
*/
/*********************************************************************************************/
void PUTLED (unsigned char d){ //LED段输入
unsignedchar i;
i= d & 0x01;
if(i== 0x00){ LED_A = 0;}
i= d & 0x02;
if(i== 0x00){ LED_B = 0;}
i= d & 0x04;
if(i== 0x00){ LED_C = 0;}
i= d & 0x08;
if(i== 0x00){ LED_D = 0;}
i= d & 0x10;
if(i== 0x00){ LED_E = 0;}
i= d & 0x20;
if(i== 0x00){ LED_F = 0;}
i= d & 0x40;
if(i== 0x00){ LED_G = 0;}
i= d & 0x80;
if(i== 0x00){ LED_DP = 0;}
}
/*******************************************************************/
void dis_off(void){
LED_A =1;//设置LED点阵屏连接的I/O口
LED_B =1;//设置LED点阵屏连接的I/O口
LED_C =1;//设置LED点阵屏连接的I/O口
LED_D =1;//设置LED点阵屏连接的I/O口
LED_E =1;//设置LED点阵屏连接的I/O口
LED_F =1;//设置LED点阵屏连接的I/O口
LED_G =1;//设置LED点阵屏连接的I/O口
LED_DP =1;//设置LED点阵屏连接的I/O口
LED1= 0 ;
LED2= 0;
LED3=0 ;
LED4= 0 ;
}
/*******************************************************************/
void displayLED(void){
dis_off();
PUTLED(tab1[display[3]]);//显示百位
LED1=1;
Delay(150);
dis_off();
PUTLED(tab1[display[2]]);//显示十位
LED2=1;
Delay(150);
dis_off();
PUTLED(0x7f&tab1[display[1]]);//显示个位和小数点
LED3=1;
Delay(150);
dis_off();
PUTLED(tab1[display[0]]);//显示小数位
LED4=1;
Delay(150);
dis_off();
}
/*******************************************************************/
/*初始化ds1820 */
/*******************************************************************/
Init_DS18B20(void)
{
DQ = 1 ; //DQ复位
Delay(8) ; //稍做延时
DQ = 0 ; //单片机将DQ拉低
Delay(90) ; //精确延时大于 480us
DQ = 1 ; //拉高总线
Delay(8) ;
presence = DQ ; //如果=0则初始化成功 =1则初始化失败
Delay(100) ;
DQ = 1 ;
return(presence) ; //返回信号,0=presence,1=no presence
}
/* 读一个字节 */
/*******************************************************************/
ReadOneChar(void)
{
unsigned char i = 0 ;
unsigned char dat = 0 ;
for (i = 8 ; i > 0 ; i--)
{
DQ = 0 ; // 给脉冲信号
dat >>= 1 ;
DQ = 1 ; // 给脉冲信号
if(DQ)
dat |= 0x80 ;
Delay(4) ;
}
return (dat) ;
}
/* 写一个字节 */
/*******************************************************************/
WriteOneChar(unsigned char dat)
{
unsigned char i = 0 ;
for (i = 8 ; i > 0 ; i--)
{
DQ = 0 ;
DQ = dat&0x01 ;
Delay(5) ;
DQ = 1 ;
dat>>=1 ;
}
}
/* 读取温度 */
/*******************************************************************/
Read_Temperature(void)
{
Init_DS18B20() ;
WriteOneChar(0xCC) ; // 跳过读序号列号的操作
WriteOneChar(0x44) ; // 启动温度转换
Init_DS18B20() ;
WriteOneChar(0xCC) ; //跳过读序号列号的操作
WriteOneChar(0xBE) ; //读取温度寄存器
temp_data[0] = ReadOneChar() ; //温度低8位
temp_data[1] = ReadOneChar() ; //温度高8位
}
/* 数据转换与温度显示 */
/*******************************************************************/
Disp_Temperature()
{
display[4]=temp_data[0]&0x0f ;
display[0]=ditab[display[4]] ; //查表得小数位的值
display[4]=((temp_data[0]&0xf0)>>4)|((temp_data[1]&0x0f)<<4);
display[3]=display[4]/100 ;
display[1]=display[4]%100 ;
display[2]=display[1]/10 ;
display[1]=display[1]%10 ;
if(display[3]==0x00) //高位为0,不显示
{
display[3]=16 ;
if(display[2]==0x00) //次高位为0,不显示
display[2]=16 ;
}
}
/* 主函数 */
/************************************/
void main()
{ P0M1=0x00;
P0M0=0x80;
P1M1=0x00;
P1M0=0x80;
P2M1=0x00;
P2M0=0x80;
P4M1=0x00;
P4M0=0x40;
P4SW= 0xff; //启动P4接口
while(1)
{
Read_Temperature() ;
Disp_Temperature() ;
displayLED();
}
}
好了,帖子写到这里,基本上算是告一段落了,我们描述了DS18B20测温芯片的封装、管脚定义、电路连接方式、内部寄存器的结构和数据格式、通信时隙和功能/控制指令,最后希望这篇帖子可以帮助到正在或者将要使用到DS18B20测温芯片的坛友,谢谢大家!
|