s3c2440实时时钟的应用


实时时钟(RTC)的主要功能是在系统掉电的情况下,利用后备电源使时钟继续运行,从而不会丢失时间信息。

s3c2440内部集成了RTC模块,而且用起来也十分简单。其内部的寄存器BCDSEC,BCDMIN,BCDHOUR,BCDDAY,BCDDATE,BCDMON和BCDYEAR分别存储了当前的秒,分,小时,星期,日,月和年,表示时间的数值都是BCD码。这些寄存器的内容可读可写,并且只有在寄存器RTCCON的第0位为1时才能进行读写操作。为了防止误操作,当不进行读写时,要把该位清零。当读取这些寄存器时,能够获知当前的时间;当写入这些寄存器时,能够改变当前的时间。另外需要注意的是,因为有所谓的“一秒误差”,因此当读取到的秒为0时,需要重新再读取一遍这些寄存器的内容,才能保证时间的正确。

下面的程序就是一个简单的RTC的例子。系统上电,或按键时,会在LCD上显示当前时间的全部信息。并且可以通过UART,在PC机上改变RTC,它们之间的协议为:当PC机发送0xAA时,表示其后的7个数据分别为年、月、日、星期、小时、分和秒,在s3c2440接收到这些数据后,发送0xAA以示确认;当PC机发送其他数据时,对于s3c2440来说没有任何意义,s3c2440只是把该数据再传回给PC机。为了简洁和突出重点,在下面程序中,会省略掉以前文章中介绍过的内容,如不清楚,请翻看本博客中的其他相关内容。 

unsigned char date_buffer[7];              //分别表示年、月、日、星期、小时、分、秒

unsigned char flag;        //用于表示更新LCD显示时间的标志

 

//获取时间

void get_date(void)

{

       rRTCCON = 1;

 

       date_buffer[0] = rBCDYEAR  ;         //年

       date_buffer[1] = rBCDMON  ;           //月

       date_buffer[2] = rBCDDATE  ;          //日 

       date_buffer[3] = rBCDDAY  ;            //星期

       date_buffer[4] = rBCDHOUR  ;         //小时

       date_buffer[5] = rBCDMIN  ;            //分

       date_buffer[6] = rBCDSEC  ;            //秒

       //当秒为0时,重新读取

       if(date_buffer[6]==0)

       {

              date_buffer[0] = rBCDYEAR  ;         //年

              date_buffer[1] = rBCDMON  ;           //月

              date_buffer[2] = rBCDDATE  ;          //日 

              date_buffer[3] = rBCDDAY  ;            //星期

              date_buffer[4] = rBCDHOUR  ;         //小时

              date_buffer[5] = rBCDMIN  ;            //分

              date_buffer[6] = rBCDSEC  ;            //秒

       }

 

       rRTCCON = 0;

}

 

//设置时间

void set_date(void)

{

       rRTCCON = 1 ;

 

       rBCDYEAR = date_buffer[0] ;            //年

       rBCDMON  = date_buffer[1] ;           //月

       rBCDDATE = date_buffer[2] ;             //日 

       rBCDDAY  = date_buffer[3] ;            //星期

       rBCDHOUR = date_buffer[4] ;            //小时

       rBCDMIN  = date_buffer[5] ;                   //分

       rBCDSEC  = date_buffer[6] ;            //秒

      

       rRTCCON = 0 ;

}

 

//在LCD上显示时间

void show_date(void)

{

       unsigned char String[ ]="当前时间:";

       unsigned char String_Y[ ]="年";

       unsigned char String_M[ ]="月";

       unsigned char String_D[ ]="日";

      

       unsigned char ToWeek[ ][7] = {"星期日","星期一","星期二","星期三","星期四","星期五","星期六"};

 

       unsigned char qh,wh;

       const unsigned char *mould;

       int length = sizeof(String);

       int k,xx;

 

       get_date();             //获取时间

 

//在LCD上写“当前时间:”这几个字

       for(k=0,xx=0;k<length-1;k++)

{

              qh=String[k]-0xa0;

              wh=String[k+1]-0xa0;

              mould = & __HZK[ (  ( qh - 1 )*94 + wh- 1 )*32 ];

              Draw_Text16(30+xx,50,0x0f,mould);

              xx+=16;

              k++;

}

 

       //在下一行显示具体的时间,形式如:2010年04月03日星期六 21:52:28

       //年

       //人为加上“20”

       mould = & __ASCII['2'*16];

Draw_ASCII(30,70,0x0,mould);

       mould = & __ASCII['0'*16];

Draw_ASCII(38,70,0x0,mould);

      

       qh=(date_buffer[0]>>4)+48;

wh=(date_buffer[0]&0x0f)+48;

mould = & __ASCII[qh*16];

       Draw_ASCII(46,70,0x0,mould);

mould = & __ASCII[wh*16];

       Draw_ASCII(54,70,0x0,mould);

   

       qh=String_Y[0]-0xa0;

       wh=String_Y[1]-0xa0;

       mould = & __HZK[ (  ( qh - 1 )*94 + wh- 1 )*32 ];

       Draw_Text16(62,70,0x0f,mould);

      

       //月

qh=(date_buffer[1]>>4)+48;

       wh=(date_buffer[1]&0x0f)+48;

mould = & __ASCII[qh*16];

       Draw_ASCII(78,70,0x0,mould);

mould = & __ASCII[wh*16];

       Draw_ASCII(86,70,0x0,mould);

   

       qh=String_M[0]-0xa0;

       wh=String_M[1]-0xa0;

       mould = & __HZK[ (  ( qh - 1 )*94 + wh- 1 )*32 ];

       Draw_Text16(94,70,0x0f,mould);

      

       //日

       qh=(date_buffer[2]>>4)+48;

wh=(date_buffer[2]&0x0f)+48;

       mould = & __ASCII[qh*16];

Draw_ASCII(110,70,0x0,mould);

       mould = & __ASCII[wh*16];

       Draw_ASCII(118,70,0x0,mould);

   

       qh=String_D[0]-0xa0;

       wh=String_D[1]-0xa0;

       mould = & __HZK[ (  ( qh - 1 )*94 + wh- 1 )*32 ];

       Draw_Text16(126,70,0x0f,mould);

 

       //星期

       for(k=0,xx=0;k<7-1;k++)

{

              qh=ToWeek[date_buffer[3]][k]-0xa0;

              wh=ToWeek[date_buffer[3]][k+1]-0xa0;

              mould = & __HZK[ (  ( qh - 1 )*94 + wh- 1 )*32 ];

              Draw_Text16(142+xx,70,0x0f,mould);

              xx+=16;

              k++;

}

   

//小时

       qh=(date_buffer[4]>>4)+48;

wh=(date_buffer[4]&0x0f)+48;

       mould = & __ASCII[qh*16];

Draw_ASCII(194,70,0x0,mould);

       mould = & __ASCII[wh*16];

Draw_ASCII(202,70,0x0,mould);

   

       mould = & __ASCII[ ':'*16 ];

       Draw_ASCII(210,70,0x0f,mould);

      

       //分

       qh=(date_buffer[5]>>4)+48;

wh=(date_buffer[5]&0x0f)+48;

       mould = & __ASCII[qh*16];

Draw_ASCII(218,70,0x0,mould);

mould = & __ASCII[wh*16];

       Draw_ASCII(226,70,0x0,mould);

   

       mould = & __ASCII[ ':'*16 ];

       Draw_ASCII(234,70,0x0f,mould);

      

       //秒

       qh=(date_buffer[6]>>4)+48;

wh=(date_buffer[6]&0x0f)+48;

       mould = & __ASCII[qh*16];

Draw_ASCII(242,70,0x0,mould);

       mould = & __ASCII[wh*16];

Draw_ASCII(250,70,0x0,mould);

}

 

//按键中断,用于更新LCD上显示的时间

void __irq Key4_ISR(void)

{

       rSRCPND = rSRCPND | 0x1;

       rINTPND = rINTPND | 0x1;

       flag=1;           //置标志

}

 

//UART中断,用于修改时间

void __irq uartISP(void)

{

       char ch;

       static char temp;

       static char count;

       rSUBSRCPND |= 0x3;

       rSRCPND |= 0x1<<28;

       rINTPND |= 0x1<<28;

      

       if(rUTRSTAT0 & 1)      //接收数据处理部分

       {

      

              ch = rURXH0;              //接收字节数据

              if(ch==0xaa&&temp==0)     //表示接收到修改时间的命令

              {

                     temp=1;

                     count=0;               

              }

              else if(temp==1)           //接收时间数据

              {

                     date_buffer[count]=ch;         //依次存入时间数组内

                     count++;

                     if(count==7)          //7个时间数据全部接收完毕

                     {

                            rUTXH0=0xaa;             //发送0xAA

                            set_date();             //设置时间

                            count=0;

                            temp=0;

                            flag=1;           //更新LCD上显示的时间

                     }

              }

              else

                     rUTXH0=ch;         //不是接收时间的命令

             

       }

}

 

void Main(void)

{

//初始化寄存器,内容省略

……

       Brush_Background(0xffffff);        //LCD背景为白色

       show_date();          //在LCD上显示时间

  

       flag=0;           //清标志

 

       while(1)

       {

              if(flag)           //需要更新LCD上的显示时间

              {

              Brush_Background(0xffffff);

                     show_date();

                     flag=0;           //清标志

              }        

}

}

相关内容