DS18B20 是美信公司的一款温度传感器,单片机可以通过 1-Wire 协议与 DS18B20 进行通信,最终将温度读出。1-Wire 总线的硬件接口很简单,只需要把 DS18B20 的数据引脚和单片机的一个 IO 口接上就可以了。硬件的简单,随之而来的,就是软件时序的复杂。1-Wire总线的时序比较复杂,很多同学在这里独立看时序图都看不明白,所以这里还要带着大家来研究 DS18B20 的时序图。我们先来看一下 DS18B20 的硬件原理图,如图 16-12 所示。
有的同学还是不能够彻底理解,程序列出来逐句解释。首先,由于 DS18B20 时序要求非常严格,所以在操作时序的时候,为了防止中断干扰总线时序,先关闭总中断。然后第一步,拉低 DS18B20 这个引脚,持续 500us;第二步,延时 60us;第三步,读取存在脉冲,并且等待存在脉冲结束。
bit Get18B20Ack(){
bit ack;
EA = 0; //禁止总中断
IO_18B20 = 0; //产生 500us 复位脉冲
DelayX10us(50);
IO_18B20 = 1;
DelayX10us(6); //延时 60us
ack = IO_18B20; //读取存在脉冲
while(!IO_18B20); //等待存在脉冲结束
EA = 1; //重新使能总中断
return ack;
}
很多同学对第二步不理解,时序图上明明是 DS18B20 等待 15us 到 60us,为什么要延时60us 呢?举个例子,妈妈在做饭,告诉你大概 5 分钟到 10 分钟饭就可以吃了,那么我们什么时候去吃,能够绝对保证吃上饭呢?很明显,10 分钟以后去吃肯定可以吃上饭。同样的道理,DS18B20 等待大概是 15us 到 60us,我们要保证读到这个存在脉冲,那么 60us 以后去读肯定可以读到。当然,不能延时太久,太久,超过 75us,就可能读不到了,为什么是 75us,大家自己思考一下。可以看出来,DS18B20 的时序比较严格,写的过程中最好不要有中断打断,但是在两个“位”之间的间隔,是大于 1 小于无穷的,那在这个时间段,我们是可以开中断来处理其它程序的。发送即写入一个字节的数据程序如下。
void Write18B20(unsigned char dat){
unsigned char mask;
EA = 0; //禁止总中断
for (mask=0x01; mask!=0; mask<<=1){ //低位在先,依次移出 8 个 bit
IO_18B20 = 0; //产生 2us 低电平脉冲
_nop_();
_nop_();
if ((mask&dat) == 0){ //输出该 bit 值
IO_18B20 = 0;
}else{
IO_18B20 = 1;
}
DelayX10us(6); //延时 60us
IO_18B20 = 1; //拉高通信引脚
}
EA = 1; //重新使能总中断
}
当要读取 DS18B20 的数据的时候,我们的单片机首先要拉低这个引脚,并且至少保持1us 的时间,然后释放引脚,释放完毕后要尽快读取。从拉低这个引脚到读取引脚状态,不能超过 15us。大家从图 16-18 可以看出来,主机采样时间,也就是 MASTER SAMPLES,是在 15us 之内必须完成的,读取一个字节数据的程序如下。
unsigned char Read18B20({
unsigned char dat;
unsigned char mask;
EA = 0; //禁止总中断
for (mask=0x01; mask!=0; mask<<=1){ //低位在先,依次采集 8 个 bit
IO_18B20 = 0; //产生 2us 低电平脉冲
_nop_();
_nop_();
IO_18B20 = 1; //结束低电平脉冲,等待 18B20 输出数据
_nop_(); //延时 2us
_nop_();
if (!IO_18B20){ //读取通信引脚上的值
dat &= ~mask;
}else{
dat |= mask;
}
DelayX10us(6); //再延时 60us
}
EA = 1; //重新使能总中断
return dat;
}
DS18B20 所表示的温度值中,有小数和整数两部分。常用的带小数的数据处理方法有两种,一种是定义成浮点型直接处理,第二种是定义成整型,然后把小数和整数部分分离出来,在合适的位置点上小数点即可。我们在程序中使用的是第二种方法,下面我们就写一个程序,将读到的温度值显示在 1602 液晶上,并且保留一位小数位。
/***************************DS18B20.c 文件程序源代码****************************/
#include
#include
sbit IO_18B20 = P3^2; //DS18B20 通信引脚
/* 软件延时函数,延时时间(t*10)us */
void DelayX10us(unsigned char t){
do {
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
_nop_();
} while (--t);
}
/* 复位总线,获取存在脉冲,以启动一次读写操作 */
bit Get18B20Ack(){
bit ack;
EA = 0; //禁止总中断
IO_18B20 = 0; //产生 500us 复位脉冲
DelayX10us(50);
IO_18B20 = 1;
DelayX10us(6); //延时 60us
ack = IO_18B20; //读取存在脉冲
while(!IO_18B20); //等待存在脉冲结束
EA = 1; //重新使能总中断
return ack;
}
/* 向 DS18B20 写入一个字节,dat-待写入字节 */
void Write18B20(unsigned char dat){
unsigned char mask;
EA = 0; //禁止总中断
for (mask=0x01; mask!=0; mask<<=1){ //低位在先,依次移出 8 个 bit
IO_18B20 = 0; //产生 2us 低电平脉冲
_nop_();
_nop_();
if ((mask&dat) == 0){ //输出该 bit 值
IO_18B20 = 0;
}else{
IO_18B20 = 1;
}
}
DelayX10us(6); //延时 60us
IO_18B20 = 1; //拉高通信引脚
EA = 1; //重新使能总中断
}
/* 从 DS18B20 读取一个字节,返回值-读到的字节 */
unsigned char Read18B20(){
unsigned char dat;
unsigned char mask;
EA = 0; //禁止总中断
for (mask=0x01; mask!=0; mask<<=1){ //低位在先,依次采集 8 个 bit
IO_18B20 = 0; //产生 2us 低电平脉冲
_nop_();
_nop_();
IO_18B20 = 1; //结束低电平脉冲,等待 18B20 输出数据
_nop_(); //延时 2us
_nop_();
if (!IO_18B20){ //读取通信引脚上的值
dat &= ~mask;
}else{
dat |= mask;
}
DelayX10us(6); //再延时 60us
}
EA = 1; //重新使能总中断
return dat;
}
/* 启动一次 18B20 温度转换,返回值-表示是否启动成功 */
bit Start18B20(){
bit ack;
ack = Get18B20Ack(); //执行总线复位,并获取 18B20 应答
if (ack == 0){ //如 18B20 正确应答,则启动一次转换
Write18B20(0xCC); //跳过 ROM 操作
Write18B20(0x44); //启动一次温度转换
}
return ~ack; //ack==0 表示操作成功,所以返回值对其取反
}
/* 读取 DS18B20 转换的温度值,返回值-表示是否读取成功 */
bit Get18B20Temp(int *temp){
bit ack;
unsigned char LSB, MSB; //16bit 温度值的低字节和高字节
ack = Get18B20Ack(); //执行总线复位,并获取 18B20 应答
if (ack == 0){ //如 18B20 正确应答,则读取温度值
Write18B20(0xCC); //跳过 ROM 操作
Write18B20(0xBE); //发送读命令
LSB = Read18B20(); //读温度值的低字节
MSB = Read18B20(); //读温度值的高字节
*temp = ((int)MSB << 8) + LSB; //合成为 16bit 整型数
}
return ~ack; //ack==0 表示操作应答,所以返回值为其取反值
}
(此处省略,可参考之前章节的代码)
/*****************************main.c 文件程序源代码******************************/
#include
bit flag1s = 0; //1s 定时标志
unsigned char T0RH = 0; //T0 重载值的高字节
unsigned char T0RL = 0; //T0 重载值的低字节
void ConfigTimer0(unsigned int ms);
unsigned char IntToString(unsigned char *str, int dat);
extern bit Start18B20();
extern bit Get18B20Temp(int *temp);
extern void InitLcd1602();
extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);
void main(){
bit res;
int temp; //读取到的当前温度值
int intT, decT; //温度值的整数和小数部分
unsigned char len;
unsigned char str[12];
EA = 1; //开总中断
ConfigTimer0(10); //T0 定时 10ms
Start18B20(); //启动 DS18B20
InitLcd1602(); //初始化液晶
while (1){
if (flag1s){ //每秒更新一次温度
flag1s = 0;
res = Get18B20Temp(&temp); //读取当前温度
if (res){ //读取成功时,刷新当前温度显示
intT = temp >> 4; //分离出温度值整数部分
decT = temp & 0xF; //分离出温度值小数部分
len = IntToString(str, intT); //整数部分转换为字符串
str[len++] = '.'; //添加小数点
decT = (decT*10) / 16; //二进制的小数部分转换为 1 位十进制位
str[len++] = decT + '0'; //十进制小数位再转换为 ASCII 字符
while (len < 6){ //用空格补齐到 6 个字符长度
str[len++] = ' ';
}
str[len] = '\0'; //添加字符串结束符
LcdShowStr(0, 0, str); //显示到液晶屏上
}else{ //读取失败时,提示错误信息
LcdShowStr(0, 0, "error!");
}
Start18B20(); //重新启动下一次转换
}
}
}
/* 整型数转换为字符串,str-字符串指针,dat-待转换数,返回值-字符串长度 */
unsigned char IntToString(unsigned char *str, int dat){
signed char i = 0;
unsigned char len = 0;
unsigned char buf[6];
if (dat < 0){ //如果为负数,首先取绝对值,并在指针上添加负号
dat = -dat;
*str++ = '-';
len++;
}
do { //先转换为低位在前的十进制数组
buf[i++] = dat % 10;
dat /= 10;
} while (dat > 0);
len += i; //i 最后的值就是有效字符的个数
while (i-- > 0){ //将数组值转换为 ASCII 码反向拷贝到接收指针上
*str++ = buf[i] + '0';
}
*str = '\0'; //添加字符串结束符
return len; //返回字符串长度
}
/* 配置并启动 T0,ms-T0 定时时间 */
void ConfigTimer0(unsigned int ms){
unsigned long tmp; //临时变量
tmp = 11059200 / 12; //定时器计数频率
tmp = (tmp * ms) / 1000; //计算所需的计数值
tmp = 65536 - tmp; //计算定时器重载值
tmp = tmp + 12; //补偿中断响应延时造成的误差
T0RH = (unsigned char)(tmp>>8); //定时器重载值拆分为高低字节
T0RL = (unsigned char)tmp;
TMOD &= 0xF0; //清零 T0 的控制位
TMOD |= 0x01; //配置 T0 为模式 1
TH0 = T0RH; //加载 T0 重载值
TL0 = T0RL;
ET0 = 1; //使能 T0 中断
TR0 = 1; //启动 T0
}
/* T0 中断服务函数,完成 1 秒定时 */
void InterruptTimer0() interrupt 1{
static unsigned char tmr1s = 0;
TH0 = T0RH; //重新加载重载值
TL0 = T0RL;
tmr1s++;
if (tmr1s >= 100){ //定时 1s
tmr1s = 0;
flag1s = 1;
}
}
太长不看版
在给出类似于DS18B20.h 和DS18B20.c之类的驱动之后,可以直接使用里面包含的初始化、写一个字节和读一个字节的函数,只要完成温度检测、读温度、结果处理和显示这三部分函数就可以了。
温度检测函数
void TempDetect()
{
Init(); //初始化,即单片机发出信号并接收应答
Delay1ms(1); //延时1ms
WriteByte(0xcc); //如果总线上只有一个器件,跳过读ROM指令
WriteByte(0x44); //写温度转换指令
Delay1ms(100); //温度转换完成,貌似不加也可以
}
void TempRead()
{
Init();
Delay1ms(1);
WriteByte(0xcc);
WriteByte(0xbe); //读温度指令
}
温度提取及显示
unsigned int TempExtraction()
{
unsigned char temph,templ;
unsigned int temp;
templ = ReadByte(); //先读低八位
temph = ReadByte(); //再读高八位
temp = temph;
temp <<= 8; //高八位左移八位再与低八位或,相当于合成2bytes 的数据
temp |= templ;
return temp;
}
void TempDisplay(uint temp)
{
float tempf;
unsigned int result;
unsigned char bai,shi,ge,shifen,baifen;
if(temp<0)
tempf=(~(temp-1))*0.0625; //如果温度是负数,读到的结果是反码,应该-1取反再乘0.0625,后续显示记得前面加0x40负号
else tempf=temp*0.0625;
result=tempf*100+0.5; //result为整数,相当于对tempf取整数部分,保留两位小数等同于乘100后取整数部分。而对于小数点第三位,由于取整默认直接截掉小数部分,所以要加0.5,这样当第三位大于等于0.5时整数部分进1位,相当于四舍五入效果
bai=result/10000; //提取各位,注意前面乘了100所以最高位应该除以10000
shi=result%10000/1000;
ge=result%1000/100;
shifen=result%100/10;
baifen=result%10;
display(num[ge]|0x80) //个位要注意加小数点,即个位数字或上0x80即可
}