久久ER99热精品一区二区-久久精品99国产精品日本-久久精品免费一区二区三区-久久综合九色综合欧美狠狠

新聞中心

EEPW首頁 > 嵌入式系統 > 設計應用 > s3c2440實時時鐘中斷

s3c2440實時時鐘中斷

作者: 時間:2016-11-19 來源:網絡 收藏
s3c2440實時時鐘(RTC)中,定義了兩個中斷源:報警中斷和時間節拍中斷。前面有網友問到了這兩個中斷的用法,最近我抽出時間對這兩個中斷研究了一番,發現這兩個中斷都很實用。現在就給大家介紹一下它們的用法。

時間節拍中斷,顧名思義,就像一個節拍器,可以等時性的控制節拍。因此它類似于定時器中斷。但時間節拍中斷是毫秒級的,而定時器中斷可以達到微秒,甚至更小級別。時間節拍中斷的周期公式為:(n+1)÷128,單位是秒,即每隔這么長時間,會中斷一次。其中n的值為1~127,它存儲在寄存器TICNT的低6位中,當寄存器TICNT的第7位被置1時,表示開啟時間節拍中斷,這時n遞減,當減為0時,進入時間節拍中斷。

報警中斷可以實現當實時時間達到預置的時間后,引起報警。預置的時間是存儲在報警時間數據寄存器中的,包括ALMYEAR(年)、ALMMON(月)、ALMDATE(日)、ALMHOUR(小時)、ALMMIN(分)和ALMSEC(秒)。而如何報警,是由報警控制寄存器RTCALM控制的。它的第6位置1表示全局報警,而第5位到第0位置1分別表示年、月、日、小時、分和秒報警。比如,我們想要在2010年4月5日22時30分0秒報警,那么把這個時間分別存儲到相應的報警時間數據寄存器中,然后設置RTCALM為0x7F,這樣當實時時鐘到達這個時刻時,會引起報警中斷;又比如我們想要系統具有鬧鐘的功能,讓它每天早上6點提醒我們起床,那么我們可以設置ALMHOUR為6,RTCALM為0x44。如果我們只想讓系統在4月份的時候提醒我們6點起床,那該怎么辦呢?這個問題對于s3c2440來說就是小菜一碟,只要我們再在ALMMON里寫入4,然后把RTCALM改為0x54即可。總之,就是系統根據RTCALM所置1的相應位來比較相對應的當前時間與報警時間數據寄存器中的值,如果相等就進入中斷。

我們對上一篇的程序進行改寫,加入報警中斷和時間節拍中斷。PC機通過UART不僅可以對s3c2440的實時時鐘進行修改,還可以設置報警時間。其中設置報警時間的通信協議與設置實時時鐘的相似,即:第一個字節為0xBB,表示命令,后面的6個字節分別是設置報警時間的年、月、日、小時、分和秒,最后一個字節用于設置RTCALM。當報警時間到時,我們利用時間節拍中斷來控制LED閃爍,閃爍15秒后自動停止,也可以通過一個按鍵來中止LED閃爍。下面的程序只列出了主要的部分:

…………
unsigned char alarm_buffer[7];//報警緩存數組
…………

//設置報警時間
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];//報警控制
}

//按鍵外部中斷,用于禁止時間節拍中斷,中止LED閃爍
void __irq Key1_ISR(void)
{
rSRCPND = rSRCPND | (0x1<<1);
rINTPND = rINTPND | (0x1<<1);

rGPBDAT = 0x1e0;//LED滅
rTICNT = 0x0;//禁止時間節拍中斷
}

//UART中斷,與上一篇文章中的相關內容相比,進行了改寫和完善
void __irq uartISR(void)
{
char ch;
static char command;
static char count;
rSUBSRCPND |= 0x3;
rSRCPND |= 0x1<<28;
rINTPND |= 0x1<<28;

if(rUTRSTAT0 & 1) //接收數據處理部分
{

ch = rURXH0; //接收字節數據
if(command==0)//判斷命令信息
{
switch(ch)
{
case 0xaa://設置實時時鐘時間
command = 0xaa;
count=0;
break;
case 0xbb://設置報警時間
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;//開啟時間節拍中斷,周期為500毫秒
}

//時間節拍中斷,用于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;//禁止時間節拍中斷
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;
}
}
}


評論


技術專區

關閉