专注电子技术学习与研究
当前位置:单片机教程网 >> MCU设计实例 >> 浏览文章

MSP430与DS18B20数码管显示(中断法)

作者:我行天下   来源:本站原创   点击数:  更新时间:2014年03月30日   【字体:

#include <msp430x14x.h>
typedef unsigned char uchar;
typedef unsigned int  uint;
/*****18B20部分的接口定义********/
#define DQ1 P1OUT |= BIT6
#define DQ0 P1OUT &= ~BIT6
#define DQ_in   P1DIR &= ~BIT6
#define DQ_out  P1DIR |= BIT6
#define DQ_val  (P1IN & BIT6)
/*****数码管部分的接口定义********/
#define wei_h P5OUT|= BIT5
#define wei_l P5OUT&= ~BIT5
#define duan_l   P6OUT &= ~BIT6
#define duan_h  P6OUT |= BIT6
//数码管七段码;0--f
uchar table[16] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
                      0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
uchar table1[16] = {0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,
0x87,0xff,0xef,0xf7,0xfc,0xb9,0xde,0xf9,0xf1};//有点
uchar tflag,num=0 ;
int tvalue;
uchar  disdata[4];
/***********18B20部分程序******************/
/*******************************************
函数名称:DelayNus
功    能:实现N个微秒的延时
参    数:n--延时长度
返回值  :无
说明    :定时器A的计数时钟是1MHz,CPU主频8MHz
          所以通过定时器延时能够得到极为精确的
          us级延时
********************************************/
void DelayNus(uint n)
{
    CCR0 = n;
    TACTL |= MC_1;           //增计数到CCR0
    while(!(TACTL & BIT0));   //等待
    TACTL &= ~MC_1;          //停止计数
    TACTL &= ~BIT0;          //清除中断标志
}
/*******************************************
函数名称:Init_18B20
功    能:对DS18B20进行复位操作
参    数:无
返回值  :初始化状态标志:1--失败,0--成功
********************************************/
uchar Init_18B20(void)
{
    uchar Error;
  
    DQ_out;
    _DINT();
    DQ0;
    DelayNus(500);
    DQ1;
    DelayNus(55);
    DQ_in;
    _NOP();
    if(DQ_val)    
    {
        Error = 1;          //初始化失败
    }
    else
    {
        Error = 0;          //初始化成功
    }
    DQ_out;
    DQ1;
    _EINT();
  
    DelayNus(400);
  
    return Error;//此处如果 Error = 1,后面就会出现死循环,表示18B20可能坏了
}
/*******************************************
函数名称:Write_18B20
功    能:向DS18B20写入一个字节的数据
参    数:wdata--写入的数据
返回值  :无
********************************************/
void Write_18B20(uchar wdata)
{
    uchar i;
  
    _DINT();
    for(i = 0; i < 8;i++)
    {
        DQ0;
        DelayNus(6);            //延时6us
        if(wdata & 0X01)    DQ1;
        else                DQ0;
        wdata >>= 1;
        DelayNus(50);           //延时50us
        DQ1;
        DelayNus(10);           //延时10us
    }
    _EINT();
}
/*******************************************
函数名称:Read_18B20
功    能:从DS18B20读取一个字节的数据
参    数:无
返回值  :读出的一个字节数据
********************************************/
uchar Read_18B20(void)
{
    uchar i;
    uchar temp = 0;
  
    _DINT();
    for(i = 0;i < 8;i++)
    {
        temp >>= 1;
        DQ0;
        DelayNus(6);            //延时6us
        DQ1;
        DelayNus(8);            //延时9us
        DQ_in;
        _NOP();
        if(DQ_val)   temp |= 0x80;
        DelayNus(45);           //延时45us
        DQ_out;
        DQ1;
        DelayNus(10);           //延时10us
    }
    _EINT();
  
    return  temp;
}
 
uint Do1Convert(void)
{
    uchar i;
    uchar temp_low;
    uint  temp;
    do
    {
        i = Init_18B20();
    }
    while(i);
    //此处的i就是等于前面的Error,Error = 1,就会出现死循环,表示18B20可能坏了
    Write_18B20(0xcc);//送跳过读取产品ID号命令
 
     Write_18B20(0x44);//发送温度转换命令
    for(i = 20;i > 0;i--)
        DelayNus(60000); //延时800ms以上
    do
    {
        i = Init_18B20();
    }
    while(i);
    //此处的i就是等于前面的Error,Error = 1,就会出现死循环,表示18B20可能坏了
  
    Write_18B20(0xcc);//送跳过读取产品ID号命令
  
     Write_18B20(0xbe);//发送读ROM命令
     
    temp_low = Read_18B20();      //读低位
    temp = Read_18B20();     //读高位
    temp = (temp<<8) | temp_low;
  
     if(temp<0x0fff)
   tflag=0;
    else
   {temp=~temp+1;
  tflag=1;
   }
tvalue=temp*(0.625);//温度值扩大10倍,精确到1位小数
  return tvalue;
}
void display(int dat)
{
disdata[0]=dat/1000;
disdata[1]=dat%1000/100;
disdata[2]=dat%100/10;
disdata[3]=dat%10;
}
/****************主函数****************/
void main(void)
{
    /*下面六行程序关闭所有的IO口*/
    P5DIR = 0xff;
    P5OUT = 0xff;P1DIR = 0XFF;P1OUT = 0XFF;
    P2DIR = 0XFF;P2OUT = 0XFF;
    P3DIR = 0XFF;P3OUT = 0XFF;
    P4DIR = 0XFF;P4OUT = 0XFF;
    P5DIR = 0XFF;P5OUT = 0XFF;
    P6DIR = 0XFF;P6OUT = 0XFF;
 
    uchar i;
  
    WDTCTL=WDTPW+WDTHOLD;
   
    /*------选择系统主时钟为8MHz-------*/
    BCSCTL1 &= ~XT2OFF;                 //打开XT2高频晶体振荡器
    do
    {
        IFG1 &= ~OFIFG;                 //清除晶振失败标志
        for (i = 0xFF; i > 0; i--);     //等待8MHz晶体起振
    }
    while ((IFG1 & OFIFG));             //晶振失效标志仍然存在?
    BCSCTL2 |= SELM_2 + SELS;           //MCLK和SMCLK选择高频晶振
 
    /*P6DIR |= BIT6;P6OUT |= BIT6;        //关闭电平转换
     P5DIR |= BIT5;P5OUT |= BIT5;        //关闭电平转换
      P6DIR |= BIT7;P6OUT |= BIT7; //关闭蜂鸣器*/
   // 设置看门狗定时器,初始化控制数码管的IO
    WDTCTL = WDT_ADLY_1_9;          
    IE1 |= WDTIE;
  
    //计数时钟选择SMLK=8MHz,1/8分频后为1MHz
    TACTL |= TASSEL_2 + ID_3;
    //打开全局中断
    _EINT();
    //循环读数显示
    while(1)
    {
      display(Do1Convert());
    }
}
/*******************************************
函数名称:watchdog_timer
功    能:看门狗定时器中断服务函数,进行数码
          管动态扫描
参    数:无
返回值  :无
********************************************/
#pragma vector = WDT_VECTOR
__interrupt void watchdog_timer(void)
{
    P4OUT = table[disdata[num]];
    if(num==2) P4OUT = table1[disdata[num]]; //加上小数点 
    duan_h;
    duan_l;
    P4OUT = ~(1<<num);//送出位码
    wei_h;
    wei_l;
    num++;
    if(num == 4) num = 0;
}
 

关闭窗口

相关文章