技术热线: 4007-888-234

QLdsPIC3]DS18B20[C30+dsPIC30F6014A]

更新时间: 2019-03-23
阅读量:3102

深圳市英锐恩科技有限公司:台湾麦肯单片机(Micon MDT单片机)亚太地区A级代理商

QLdsPIC3]DS18B20[C30+dsPIC30F6014A]
//实验目的:熟悉DS18B20的使用
//转换结果送LCD1602显示(其中整数部分3位,小数点部分4位,小数点1位)
//每隔0.1S转换一次。
//硬件要求:把DS18B20插在18B20插座上
//          跳线J5接通
//          跳线J18全部接通

#include             //dsPIC30F6014标准头文件

  _FOSC(0x0ffe5);                 //XT振荡,4倍频晶振.
  _FWDT(WDT_OFF);                 //关闭看门狗定时器
  _FBORPOR(PBOR_OFF & MCLR_EN);   //掉电复位禁止,MCLR复位使能。
  _FGS(CODE_PROT_OFF);            //代码保护禁止

#define  uch unsigned char                     //给unsigned char起别名 uch
# define DQ PORTFbits.RF6                      //定义18B20数据端口
# define DQ_DIR TRISFbits.TRISF6               //定义18B20D口方向寄存器
# define DQ_HIGH() DQ_DIR =1                   //设置数据口为输入
# define DQ_LOW() DQ = 0; DQ_DIR = 0           //设置数据口为输出

#define rs LATBbits.LATB4          //定义LCD控制位(注意这里只能用LATB寄存器,不能直接用PORTB寄存器)
#define rw LATBbits.LATB5
#define e  LATBbits.LATB6
                       
unsigned char __attribute__((address(0x900))) TLV;   //采集到的温度高8位
unsigned char __attribute__((address(0x902))) THV;   //采集到的温度低8位
unsigned char __attribute__((address(0x904))) TZ;    //转换后的温度值整数部分
unsigned char __attribute__((address(0x906))) TX;    //转换后的温度值小数部分
unsigned int  __attribute__((address(0x908))) wd;    //转换后的温度值BCD码形式
unsigned char __attribute__((address(0x9a0))) loop;  //LCD显示次数计数器

unsigned char __attribute__((address(0x930))) result[8]={0,0,0,'.',0,0,0,0};

//****************延时函数**************************************
//延时函数(延时时间为(2+x)us)
void delay(  char x)
{
   unsigned char i;
   for(i=0;i<x;i++);
}

//*****************写一个字节数据函数***************************
//在电平发生改变后需要插入一段延时时间,否则LCD反应不过来。
void write(unsigned char x)
 {
  PORTD=x;                   //待显示数据送PORTD口
  delay(98);                  
  rs=1;                      //该字节数据为数据,而不是命令
  delay(98);
  rw=0;                      //此次操作为写,而不是读
  delay(98);
  e=0;                       //拉低使能信号
  delay(98);                   //保持使能信号为低一段时间
  e=1;                       //拉高使能信号,建立LCD操作所需要的上升沿
  delay(98);
 }

//******************LCD显示设置函数*****************************
//在电平发生改变后需要插入一段延时时间,否则LCD反应不过来。
void lcd_enable()
 {
   delay(98);
   rs=0;                     //该字节数据为命令,而不是数据
   delay(98);
   rw=0;                     //此次操作为写,而不是读
   delay(98);
   e=0;                      //拉低使能信号
   delay(98);                  //保持使能信号为低一段时间
   e=1;                      //拉高使能信号,建立LCD操作所需要的上升沿
   delay(98);
}

//------------------------------------------------
//系统初始化函数
void init()
{
  TRISF=0X0000;                                 //先设置18B20口方向为输出
  TRISB=0X0000;                                 //设置B口方向为输出
  TRISD=0X0000;                                 //设置D口方向为输出
  IFS0bits.T1IF=0;                              //先清除定时器中断标志位
  PR1=0XFFFF;                                   //周期定最大
  T1CON=0X0020;                                 //分频比为1:64
}

//*******************LCD初始化函数*******************************
void lcd_init()
 {
    PORTD=0X1;                 //清除显示
    lcd_enable();
    PORTD=0X38;                //8位2行5*7点阵
    lcd_enable();
    PORTD=0X0e;                //显示开,光标开,闪烁
    lcd_enable();
    PORTD=0X06;                //文字不动,光标右移
    lcd_enable();
 }
 
//*********************LCD显示函数*******************************
void display()
 {
    PORTD=0X84;                //显示首地址
    lcd_enable();
    for(loop=0;loop<8;loop++)  //共显示8位
      {
        write(result[loop]);   //查表并送显示
        delay(100);            //插入一定延时
      }
 }

//******************复位DS18B20函数*****************************
//主控制器(dsPIC30F6014A)先拉低总线480us,然后释放总线回到高电平
//18B20检测到上升沿后先等待15-60us,然后拉低总线做为复位的应答信号
//主控制器释放总线后到复位结束时间应为480US
reset(void)
{
  char presence=1;
  while(presence)
  {
    DQ_LOW() ;                                //主机拉至低电平
    delay(100);                        
    delay(100);  
    delay(100);  
    delay(100);
    delay(78);                                //以上5条共延时480us
    DQ_HIGH();                                //释放总线等电阻拉高总线,并保持15~60us
    delay(38);                                //延时40us       
    if(DQ==1) presence=1;                     //没有接收到应答信号,继续复位
    else presence=0;                          //接收到应答信号
    delay(100);                        
    delay(100);  
    delay(100);  
    delay(100);
    delay(38);                                //以上5条共延时440us
   }
  }

//****************写18b20写字节函数*****************************
//主控制器写数据1:先把总线拉低,然后在15us内释放总线
//主控制器写数据0:把总线拉低至少60us
//写一位数据至少需要60us
//两位数据之间至少延时1us
void write_byte(uch val)
{
 uch i;
 uch temp;
 for(i=8;i>0;i--)
 {
   temp=val&0x01;                            //最低位移出
   DQ_LOW();
   delay(3);                                 //保持拉低5us
   if(temp==1)  DQ_HIGH();                   //如果写1,拉高电平
   delay(58);                                //延时60us
   DQ_HIGH();
   delay(1);                                 //在两位之间插入3us延时
   val=val>>1;                               //右移一位
  }
}

//****************18b20读字节函数********************************
//主控制器把总线拉低至少1US,然后释放
//主控制器读数据1:18B20保持总线状态不变
//主控制器读数据0:18B20检测到总线拉低后继续拉低总线至少60ms
//主控制器在拉低总线后的15us读取总线上的状态
//读取1位数据至少需要60us
uch read_byte(void)
{
 uch i;
 uch value=0;                                //读出温度
 for(i=8;i>0;i--)
 {
   value>>=1;
   DQ_LOW();
   delay(1);                                //保持总线拉低3us
   DQ_HIGH();                               //拉至高电平
   delay(5);                                //释放总线后保持7us再读取数据
   if(DQ) value|=0x80;
   delay(48);                               //延时50us,保证每1位的60us延时
  }
  return(value);
}

//-------------------------------------------------
//启动温度转换函数
void get_temp()
{
int i;
DQ_HIGH();                              
reset();                                 //复位等待从机应答
write_byte(0XCC);                        //忽略ROM匹配
write_byte(0X44);                        //发送温度转化命令
for(i=0;i<8;i++)
    {
       
      delay(98);                         //确保温度转换完成所需要的时间
    }
reset();                                 //再次复位,等待从机应答
write_byte(0XCC);                        //忽略ROM匹配
write_byte(0XBE);                        //发送读温度命令
TLV=read_byte();                         //读出温度低8
THV=read_byte();                         //读出温度高8位
DQ_HIGH();                               //释放总线
TZ=(TLV>>4)|(THV<<4)&0X3f;               //温度整数部分
TX=TLV<<4;                               //温度小数部分

result[0]=(TZ/100)+0x30;                 //整数百位
result[1]=((TZ%100)/10)+0X30;            //整数十位
result[2]=((TZ%100)%10)+0X30;            //整数部分个位          
wd=0;
if (TX & 0x80) wd=wd+5000;
if (TX & 0x40) wd=wd+2500;
if (TX & 0x20) wd=wd+1250;
if (TX & 0x10) wd=wd+625;                //以上4条指令把小数部分转换为BCD码形式            
result[4]=(wd/1000)+0x30;                //十分位
result[5]=((wd%1000)/100)+0x30;          //百分位
result[6]=((wd%100)/10)+0x30;            //千分位
result[7]=(wd%10)+0x30;                  //万分位
}

//--------------------------------------------------
//主猪函数
int main(void)
{
   init();                              //调用系统初始化函数
   lcd_init();                          //LCD初始化函数
   while(1)
     {
       TMR1=0XC2F6;                     //定时器初值
       T1CONbits.TON=1;                 //启动定时器
       while(!IFS0bits.T1IF);           //等待0.1s定时到
       IFS0bits.T1IF=0;
       get_temp();                      //调用温度转换函数
       display();                       //调用结果显示函数
     }
}