s3c2440實時時鐘中斷
時間節(jié)拍中斷,顧名思義,就像一個節(jié)拍器,可以等時性的控制節(jié)拍。因此它類似于定時器中斷。但時間節(jié)拍中斷是毫秒級的,而定時器中斷可以達到微秒,甚至更小級別。時間節(jié)拍中斷的周期公式為:(n+1)÷128,單位是秒,即每隔這么長時間,會中斷一次。其中n的值為1~127,它存儲在寄存器TICNT的低6位中,當寄存器TICNT的第7位被置1時,表示開啟時間節(jié)拍中斷,這時n遞減,當減為0時,進入時間節(jié)拍中斷。
報警中斷可以實現(xiàn)當實時時間達到預(yù)置的時間后,引起報警。預(yù)置的時間是存儲在報警時間數(shù)據(jù)寄存器中的,包括ALMYEAR(年)、ALMMON(月)、ALMDATE(日)、ALMHOUR(小時)、ALMMIN(分)和ALMSEC(秒)。而如何報警,是由報警控制寄存器RTCALM控制的。它的第6位置1表示全局報警,而第5位到第0位置1分別表示年、月、日、小時、分和秒報警。比如,我們想要在2010年4月5日22時30分0秒報警,那么把這個時間分別存儲到相應(yīng)的報警時間數(shù)據(jù)寄存器中,然后設(shè)置RTCALM為0x7F,這樣當實時時鐘到達這個時刻時,會引起報警中斷;又比如我們想要系統(tǒng)具有鬧鐘的功能,讓它每天早上6點提醒我們起床,那么我們可以設(shè)置ALMHOUR為6,RTCALM為0x44。如果我們只想讓系統(tǒng)在4月份的時候提醒我們6點起床,那該怎么辦呢?這個問題對于s3c2440來說就是小菜一碟,只要我們再在ALMMON里寫入4,然后把RTCALM改為0x54即可??傊?,就是系統(tǒng)根據(jù)RTCALM所置1的相應(yīng)位來比較相對應(yīng)的當前時間與報警時間數(shù)據(jù)寄存器中的值,如果相等就進入中斷。
我們對上一篇的程序進行改寫,加入報警中斷和時間節(jié)拍中斷。PC機通過UART不僅可以對s3c2440的實時時鐘進行修改,還可以設(shè)置報警時間。其中設(shè)置報警時間的通信協(xié)議與設(shè)置實時時鐘的相似,即:第一個字節(jié)為0xBB,表示命令,后面的6個字節(jié)分別是設(shè)置報警時間的年、月、日、小時、分和秒,最后一個字節(jié)用于設(shè)置RTCALM。當報警時間到時,我們利用時間節(jié)拍中斷來控制LED閃爍,閃爍15秒后自動停止,也可以通過一個按鍵來中止LED閃爍。下面的程序只列出了主要的部分:
…………
unsigned char alarm_buffer[7];//報警緩存數(shù)組
…………
//設(shè)置報警時間
void set_alarm(void)
{
rALMYEAR = alarm_buffer[0];//年
rALMMON = alarm_buffer[1];//月
rALMDATE = alarm_buffer[2];//日
rALMHOUR = alarm_buffer[3];//小時
rALMMIN = alarm_buffer[4];//分
rALMSEC = alarm_buffer[5];//秒
rRTCALM = alarm_buffer[6];//報警控制
}
//按鍵外部中斷,用于禁止時間節(jié)拍中斷,中止LED閃爍
void __irq Key1_ISR(void)
{
rSRCPND = rSRCPND | (0x1<<1);
rINTPND = rINTPND | (0x1<<1);
rGPBDAT = 0x1e0;//LED滅
rTICNT = 0x0;//禁止時間節(jié)拍中斷
}
//UART中斷,與上一篇文章中的相關(guān)內(nèi)容相比,進行了改寫和完善
void __irq uartISR(void)
{
char ch;
static char command;
static char count;
rSUBSRCPND |= 0x3;
rSRCPND |= 0x1<<28;
rINTPND |= 0x1<<28;
if(rUTRSTAT0 & 1) //接收數(shù)據(jù)處理部分
{
ch = rURXH0; //接收字節(jié)數(shù)據(jù)
if(command==0)//判斷命令信息
{
switch(ch)
{
case 0xaa://設(shè)置實時時鐘時間
command = 0xaa;
count=0;
break;
case 0xbb://設(shè)置報警時間
command = 0xbb;
count=0;
break;
default://其余命令
command = 0;
count =0;
rUTXH0=ch;
break;
}
}
else//接收實時時鐘時間或報警時間
{
if(command == 0xaa)//實時時鐘時間
{
date_buffer[count]=ch;
count++;
if(count==7)
{
set_date();
count=0;
command=0;
flag=1;
rUTXH0=0xaa;
}
}
else if(command ==0xbb)//報警時間
{
alarm_buffer[count]=ch;
count++;
if(count==7)
{
set_alarm();
count=0;
command=0;
rUTXH0=0xbb;
}
}
}
}
}
//報警中斷
void __irq Alarm_ISR(void)
{
rSRCPND |= 0x1<<30;
rINTPND |= 0x1<<30;
rTICNT = 0xbf;//開啟時間節(jié)拍中斷,周期為500毫秒
}
//時間節(jié)拍中斷,用于LED閃爍15秒
void __irq RTCTick_ISP(void)
{
static char count;
rSRCPND |= 0x1<<8;
rINTPND |= 0x1<<8;
if(count%2==0)//LED亮0.5秒
rGPBDAT = ~0x1e0;
else//LED滅0.5秒
rGPBDAT = 0x1e0;
count++;
if(count==30)
{
rTICNT = 0x0;//禁止時間節(jié)拍中斷
rGPBDAT = 0x1e0;//LED滅
count=0;
}
}
void Main(void)
{
//初始化
…………
//中斷源
pISR_UART0 = (U32)uartISR;
pISR_EINT0 = (U32)Key4_ISR;
pISR_EINT1 = (U32)Key1_ISR;
pISR_RTC = (U32)Alarm_ISR;
pISR_TICK = (U32)RTCTick_ISP;
…………
Brush_Background(0xffffff);
show_date();
flag=0;
while(1)
{
if(flag)//顯示實時時間
{
Brush_Background(0xffffff);
show_date();
flag=0;
}
}
}
評論