×
单片机 > 单片机程序设计 > 详情

DS18B20多点温度检测系统

发布时间:2020-08-21 发布时间:
|
#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    {
    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]);
   }
  }
 }
}




『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
STM32 USB HID 键盘