DS18B20多点温度检测系统
扫描二维码
随时随地手机看文章
本文为DS18B20多点温度检测系统设计程序
#include
#include //_nop_();延时函数用
#define uchar unsigned char
#define uint unsigned int
//********************** DS18B20 指令********************* *********
#define ds18B20_READ_ROM 0x33 // 读ROM指令
#define ds18B20_MATCH_ROM 0x55 // 匹配ROM指令
#define ds18B20_SKIP_ROM 0xCC // 跳过ROM指令
#define ds18B20_SEARCH_ROM 0xF0 // 搜索ROM指令
#define ds18B20_ALARM_SEARCH 0xEC // 报警搜索指令
#define ds18B20_WRITE_SCRATCHPAD 0x4E // 写暂存寄存器指令
#define ds18B20_READ_SCRATCHPAD 0xBE // 读暂存寄存器指令
#define ds18B20_COPY_SCRATCHPAD 0x48 // 复制暂存寄存器指令
#define ds18B20_CONVERT_T 0x44 // 启动温度转换指令
#define ds18B20_RECALL_E2 0xB8 // 重新调出E2PROM的数据
sbit DQ=P0^7; //将P0.7口模拟时钟数据口
bit flag=0;
/******************************
温 度小数部分用查表法
*******************************/
uchar code ditab[16]=
{0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09};
/**************************
延 时函数
***************************/
void delay_us(unsigned int us)
{
unsigned int i;
us=us/7; //1/7是STC89C58在22.1184MHz晶振
for( i=0;i }
void delay_ms(unsigned int ms)
{
unsigned int i,j;
for( i=0;i for(j=0;j<332;j++); //332是STC89C58在22.1184MHz晶振
}
/***********读出温度函数************
输 入:无
输出:zs+xs,格式为123.4
即为浮点型数的温度格式
***********************************/
/***********18B20 复位函数**********/
void Init18b20 (void)
{
DQ=1;;_nop_();_nop_();
DQ=0; delay_us(600); //要大于480us,小于960us
DQ=1;delay_us(60); //要大于15us-- 60us
if(DQ==0)
flag = 1; //detect 1820 success!
else
flag = 0; //detect 1820 fail!
delay_us(200); //要大于60us,小于 240us
DQ = 1;
}
/**********18B20写命令函数*********
向 1-WIRE 总线上写一个字节
**********************************/
void write_byte(uchar val)
{
uchar i;
for (i=8; i>0; i--)
{
DQ=1;_nop_();_nop_();
DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us
DQ = val&0x01; //最低位移出
delay_us(60); //66us
val=val>>1; //右移一位
}
DQ = 1;
delay_us(10);
}
/*********18B20读1个字节 函数********
从总线上读取一个字节
***********************************/
uchar read_byte(void)
{
uchar i;
uchar value = 0;
for (i=8;i>0;i--)
{
DQ=1;_nop_();_nop_();
value>>=1;
DQ = 0;_nop_();_nop_();_nop_();_nop_(); //4us
DQ = 1;_nop_();_nop_();_nop_();_nop_(); //4us
if(DQ)value|=0x80;
delay_us(60); //60us
}
DQ=1;
delay_us(10);
return(value);
}
/***************************
串 口中断服务子程序
****************************/
static void com_int(void) interrupt 4
{
if (RI)
{
RI = 0;
if (SBUF == 0x0d)
{
ISP_CONTR = 0x60; // 软复位到ISP区
}
}
}
/**************************
串口发送一个字符
**************************/
void com_send_dat( unsigned char dat)
{
SBUF=dat;
while (TI== 0);
TI= 0 ;
}
/**************************
串口初始化
**************************/
void init_com( void )
{
SCON=0x50 ; //SCON: serail mode 1, 8-bit UART, enable ucvr //UART为模式1,8位数据,允许接收
TMOD|=0x20 ; //TMOD: timer 1, mode 2, 8-bit reload //定时器1为模式2,8位自动重装
TH1=0xfa ; //Baud:19200 fosc="22.1184MHz
TL1=0xfa;
PCON|=0x80; //SMOD=1;波特率加倍;
ES=1; //Enable Serial Interrupt
TR1 = 1 ; // timer 1 run
}
/**************************
搜索DS18B2064位ROM
**************************/
uchar search_rom(uchar *p)
{
uchar dat,i,j,k,n,value,number=1,number_temp=1,clash_number=0;
bit clash_flag=0;
for(n=0;n {
flag=0;
while(!flag)Init18b20(); // 复位ds18b20
write_byte(ds18B20_SEARCH_ROM); //搜索ROM命令
for(i=0; i<8;i++) //8个字节第ROM
{
dat=0;
for(j=0;j<8; j++) //8位数据
{
value=0;
for(k=0;k<2;k++) //一位数据 位,一位补码位
{
DQ=1;_nop_();_nop_();
value<<=1;
DQ = 0;_nop_();_nop_();_nop_();_nop_(); //4us
DQ = 1;_nop_();_nop_();_nop_();_nop_(); //4us
if(DQ)value|=0x01;
delay_us(60); //60us
}
switch(value) //根据回复第两位数据来发出相应操作
{
case 0x00: //总线冲突
if(n==0)
{
number++;
number_temp=number;
dat=dat>>1;
DQ = 1;_nop_();_nop_();
DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us
DQ = 0; //最低位移出
delay_us(60); //60us
}
else // 第二次以上循环时
{
clash_number++;
if(clash_number>=(number_temp-1))
{
if(!clash_flag)
{
clash_flag=1;
dat=dat>>1;
dat=dat|0x80;
DQ = 1;_nop_();_nop_();
DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us
DQ = 1; //最低位移出
delay_us(60); //60us
}
else
{
clash_flag=0;
number_temp--;
dat=dat>>1;
DQ = 1;_nop_();_nop_();
DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us
DQ = 0; //最低位移出
delay_us(60); //60us
}
}
}
break;
case 0x01:
dat=dat>>1;
DQ = 1;_nop_();_nop_();
DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us
DQ = 0; //最低位移出
delay_us(60); //66us
break;
case 0x02:
dat=dat>>1;
dat=dat|0x80;
DQ = 1;_nop_();_nop_();
DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us
DQ = 1; //最低位移出
delay_us(60); //66us
break;
case 0x03:i=8;j=8;break;
default:break;
}
}
*p++=dat;
}
}
return number;
}
/****************************
多个DS18B20测试主函数
****************************/
void main()
{
uchar i,j,dat,amount;
uchar ROM_data[16]; // 读出ROM暂存
uchar temp_data[2]; //读出温度暂存
uchar temperature[7]={0,0,0,'.',0,'?,'?}; //要发送的数据缓存
init_com();
EA=0;
while(1)
{
amount=search_rom(ROM_data);
flag=0;
while(!flag)Init18b20();
write_byte(ds18B20_SKIP_ROM); // 跳过ROM命令
write_byte(ds18B20_CONVERT_T); // 发转换命令
EA=1;
delay_ms(1000);
EA=0;
for(j=0;j {
flag=0;
while(!flag)Init18b20();
write_byte(ds18B20_MATCH_ROM); // 符合ROM命令
for(i=j*8;i<(j+1)*8;i++)
{
write_byte(ROM_da
ta[i]);
}
write_byte(ds18B20_READ_SCRATCHPAD);//发读命令
temp_data[0]=read_byte(); //温度低8位
temp_data[1]=read_byte(); //温度高8位
temperature[4]=ditab[temp_data[0]&0x0f]+0x30; // 小数位
dat=((temp_data[0]&0xf0)>>4)|((temp_data[1]&0x0f)<<4); // 整数位
for(i=0;i<3;i++)
{
temperature[2-i]=dat%10+0x30;
dat=dat/10;
}
for(i=0; i<7;i++) //串口发送温度数据
{
com_send_dat(temperature[i]);
}
}
}