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;
}
}
}