S3C2440-RTC中断
发布时间:2011-3-25 14:25
发布者:techshare
RTC(Real Time Clock)单元能在系统断电时通过系统备用电池供电,RTC能通过ARM的STRB/LDRB指令传输8位BCD数据到CPU,该数据包括时,分,秒,小时,天,月和年,RTC使用一个外部32.768KHZ的晶体也能执行报警功能。 报警功能。在掉电模式或正常工作模式下,RTC能够在指定的时间产生报警信号。在正常工作模式下,报警中断(ALMINT)被激活。在掉电模式下,电源管理苏醒信号PMWKUP也与ALMINT一样处于激活状态。RTC的报警寄存器(RTCALM)可以决定报警的使能或禁止和报警时间的设置条件。 节拍中断。RTC节拍时间用于中断请求。TICNT寄存器具有一个中断使能位,同时其中的计数值用于中断。当计数值到达0时,节拍时间中断就会触发。中断的时间间隔计算如下: 中断的间隔时间 = (n+1)/128 单位是秒 其中n的值在1"127 下面的程序实现了,按一个键,可以更新当前时间,可以通过串口发送数据修改当前时间,设定报警寄存器,报警被触发后,会触发时间节拍中断,中断时间间隔位1s,持续时间是20s,也就是间隔1s蜂鸣器响一下。按另一个键,停止节拍时间中断,停止蜂鸣器。 ![]() #include "Font_Libs.h" #include "2440addr.h" #define _ISR_STARTADDRESS 0x33ffff00 #define pISR_EINT0 (*(unsigned *)(_ISR_STARTADDRESS+0x20)) #define pISR_TICK (*(unsigned *)(_ISR_STARTADDRESS+0x40)) #define pISR_RTC (*(unsigned *)(_ISR_STARTADDRESS+0x98)) #define pISR_EINT2 (*(unsigned *)(_ISR_STARTADDRESS+0x28)) #define pISR_UART0 (*(unsigned *)(_ISR_STARTADDRESS+0x90)) //垂直同步信号的脉宽、后肩和前肩 #define VSPW 15 #define VBPD 3 #define VFPD 5 //水平同步信号的脉宽、后肩和前肩 #define HSPW 8 #define HBPD 58 #define HFPD 15 #define CLKVAL 10 #define HOZVAL 319 #define LINEVAL 239 #define PWREN 1 #define MMODE 0 #define PNRMODE 3 #define BPPMODE 13 #define INVVCLK 0 #define INVVD 0 #define INVVDEN 0 #define U32 unsigned int #define M5D(n) ((n) & 0x1fffff) #define PAGEWIDTH 320 #define OFFSIZE 0 #define LCD_XSIZE 320 #define LCD_YSIZE 240 #define SCR_XSIZE 320 #define SCR_YSIZE 240 #define INVVLINE 1 #define INVVFRAME 1 #define BPP24BL 0 #define BSWP 0 #define HWSWP 0 volatile U32 LCD_BUFFER[240][320]; unsigned char data_buffer[7] = {5*16+1,1*16+7,1*16+9,2*16+6,7,1*16+2,1*16+0}; unsigned char alarm_buffer[6] ={0,1*16+8,0,0,0,0}; unsigned char *temp; unsigned char str0[] = "当前时间为"; unsigned char str1[] = "年"; unsigned char str2[] = "月"; unsigned char str3[] = "日"; unsigned char str[][7] = {"星期日","星期一","星期二","星期三","星期四","星期五","星期六"}; U32 flag, count, t, i; void Init_LCD(){ rLCDCON1=(CLKVAL<<8)|(MMODE<<7)|(PNRMODE<<5)|(BPPMODE<<1)|0; //设置CLKVAL,VCLK=HCLK/[(CLKVAL+1)*2],决定VM的触发方式,选择显示模式和BPP模式,暂时不要开启LCD,因为还没有设置好 rLCDCON2=(VBPD<<24)|(LINEVAL<<14)|(VFPD<<6)|(VSPW); //rLCDCON2,rLCDCON3和rLCDCON4主要设置时序 rLCDCON3=(HBPD<<19)|(HOZVAL<<8)|(HFPD); rLCDCON4=(HSPW); rLCDCON5 = (BPP24BL<<12) | (INVVCLK<<10) | (INVVLINE<<9) | (INVVFRAME<<8) | (0<<7) | (INVVDEN<<6) | (PWREN<<3) |(BSWP<<1) | (HWSWP); //INVVLINE和INVVFRAME需要进行翻转,因为CPU发出的是正脉冲,LCD使用的是负脉冲,所以要改变极性,PWREN使能电源信号 rLCDSADDR1=(((U32)LCD_BUFFER>>22)<<21)|M5D((U32)LCD_BUFFER>>1); rLCDSADDR2=M5D(((U32)LCD_BUFFER+(SCR_XSIZE*SCR_YSIZE*4))>>1 ); rLCDSADDR3=PAGEWIDTH*32/16; rLCDINTMSK|=(3); rTCONSEL = 0; rGPCUP = 0x0; rGPDCON = 0xaaaaaaaa; rGPCCON = 0xaaaa02a9; rGPDUP = 0x0; rGPGUP=rGPGUP&("(1<<4))|(1<<4); rGPGCON=rGPGCON&("(3<<8))|(3<<8); rLCDCON1 |= 1; //使能数据输出和LCD控制信号 } void Paint_background(U32 c, U32 startx, U32 starty, U32 endx, U32 endy){ U32 i,j; for(j = starty; j < endy; j++) for(i = startx; i < endx; i++) LCD_BUFFER[j][ i] = c; } void SetTime(){ rRTCCON |= 0x1; rBCDSEC = data_buffer[0]; rBCDMIN = data_buffer[1]; rBCDHOUR = data_buffer[2]; rBCDDATE = data_buffer[3]; rBCDDAY = data_buffer[4]; rBCDMON = data_buffer[5]; rBCDYEAR = data_buffer[6]; rRTCCON &= 0xfe; } void GetTime(){ rRTCCON |= 0x1; data_buffer[0] = rBCDSEC; data_buffer[1] = rBCDMIN; data_buffer[2] = rBCDHOUR; data_buffer[3] = rBCDDATE; data_buffer[4] = rBCDDAY; data_buffer[5] = rBCDMON; data_buffer[6] = rBCDYEAR; if(data_buffer[0] == 0){ data_buffer[0] = rBCDSEC; data_buffer[1] = rBCDMIN; data_buffer[2] = rBCDHOUR; data_buffer[3] = rBCDDATE; data_buffer[4] = rBCDDAY; data_buffer[5] = rBCDMON; data_buffer[6] = rBCDYEAR; } rRTCCON &= 0xfe; } void Paint_text(U32 x, U32 y, U32 color, unsigned char ch[]){ int i, j, test,t = 0; for(i = 0; i < 16; i++){ test = 0x80; for(j = 0; j < 16; j++){ if(j == 8){ test = 0x80; t++; } if(ch[t] & test) LCD_BUFFER[x+i][y+j] = color; test >>= 1; } t++; } } void Paint_Ascii(U32 x, U32 y, U32 color, unsigned char ch[]){ int i, j, test; for(i = 0; i < 16; i++){ test = 0x80; for(j = 0; j < 8; j++){ if(test & ch[ i]) LCD_BUFFER[x+i][y+j] = color; test >>= 1; } } } void __irq Uart_ISR(void){ char buf; rSUBSRCPND |= 0x3; rSRCPND |= 0x1<<28; rINTPND |= 0x1<<28; if(rUTRSTAT0&0x1){ buf = rURXH0; if((buf == 0xaa) && (t == 0)){ t = 1; i = 0; rUTXH0 = 0xaa; } else{ if(t){ data_buffer[ i] = buf; i++; rUTXH0 = 0xdd; if(i == 7){ rUTXH0 = 0xbb; SetTime(); i = 0; t = 0; flag = 1; } } else{ rUTXH0 = 0xcc; } } } } //2010年12月26日星期日19:17:51 void ShowTime(){ U32 qh, wh; unsigned char s0,s1; int i, t, k; char h, l; GetTime(); //当前时间为 for(i = 0,t = 0, k = 0; i < 5; i++){ s0 = str0[t]; s1 = str0[t+1]; qh = s0-0xa0; wh = s1-0xa0; temp = & __HZK[((qh-1)*94+wh-1)*32]; Paint_text(100,k,0x0,temp); t = t + 2; k += 16; } //: temp = &__ASCII[':'*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //20 temp = &__ASCII['2'*16]; Paint_Ascii(100,k,0x0,temp); k+=8; temp = &__ASCII['0'*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //10 h = (data_buffer[6]>>4)+48; temp = &__ASCII[h*16]; Paint_Ascii(100,k,0x0,temp); k+=8; l = (data_buffer[6]&0x0f)+48; temp = &__ASCII[l*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //年 qh = str1[0]-0xa0; wh = str1[1]-0xa0; temp = & __HZK[((qh-1)*94+wh-1)*32]; Paint_text(100,k,0x0,temp); k+=16; //12 h = (data_buffer[5]>>4)+48; temp = &__ASCII[h*16]; Paint_Ascii(100,k,0x0,temp); k+=8; l = (data_buffer[5]&0xf)+48; temp = &__ASCII[l*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //月 qh = str2[0]-0xa0; wh = str2[1]-0xa0; temp = & __HZK[((qh-1)*94+wh-1)*32]; Paint_text(100,k,0x0,temp); k+=16; //26 h = (data_buffer[3]>>4)+48; temp = &__ASCII[h*16]; Paint_Ascii(100,k,0x0,temp); k+=8; l = (data_buffer[3]&0xf)+48; temp = &__ASCII[l*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //日 qh = str3[0]-0xa0; wh = str3[1]-0xa0; temp = & __HZK[((qh-1)*94+wh-1)*32]; Paint_text(100,k,0x0,temp); k+=16; //星期日 for(i = 0,t = 0; i < 3; i++){ s0 = str[0][t]; s1 = str[0][t+1]; qh = s0-0xa0; wh = s1-0xa0; temp = & __HZK[((qh-1)*94+wh-1)*32]; Paint_text(100,k,0x0,temp); t = t + 2; k += 16; } //19:17:51 //19 h = (data_buffer[2]>>4)+48; temp = &__ASCII[h*16]; Paint_Ascii(100,k,0x0,temp); k+=8; l = (data_buffer[2]&0xf)+48; temp = &__ASCII[l*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //: temp = &__ASCII[':'*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //17 h = (data_buffer[1]>>4)+48; temp = &__ASCII[h*16]; Paint_Ascii(100,k,0x0,temp); k+=8; l = (data_buffer[1]&0xf)+48; temp = &__ASCII[l*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //: temp = &__ASCII[':'*16]; Paint_Ascii(100,k,0x0,temp); k+=8; //51 h = (data_buffer[0]>>4)+48; temp = &__ASCII[h*16]; Paint_Ascii(100,k,0x0,temp); k+=8; l = (data_buffer[0]&0xf)+48; temp = &__ASCII[l*16]; Paint_Ascii(100,k,0x0,temp); k+=8; } void SetAlarm(){ rALMSEC = alarm_buffer[0]; rALMMIN = alarm_buffer[1]; rALMHOUR = alarm_buffer[2]; rALMDATE = alarm_buffer[3]; rALMMON = alarm_buffer[4]; rALMYEAR = alarm_buffer[5]; } void __irq EINT0_ISR(void){ rSRCPND |= 1; //SRCPND 通过写入数据清零,如果不清零,会反复进行请求 rINTPND |= 1; //INDPND 通过置1清零 flag = 1; } void __irq EINT2_ISR(void){ //使用__irq这个关键字定义终端服务例程,这样系统会自动为我们保留一些变量,并能在中断处理完后正确的返回 rSRCPND |= 1<<2; rINTPND |= 1<<2; rTICNT &= "(1<<7); rGPBDAT = 0xfe; //如果按键被按下,关闭节拍时间中断 } void __irq Alarm_ISR(void){ rSRCPND |= 1<<30; //SRCPND 通过写入数据清零,如果不清零,会反复进行请求 rINTPND |= 1<<30; //INDPND 通过置1清零 rTICNT = (1<<7)|(127<<0); //开启节拍时间中断,设置每1s中断一次 } void __irq Tick_ISR(void){ rSRCPND |= 1<<8; //SRCPND 通过写入数据清零,如果不清零,会反复进行请求 rINTPND |= 1<<8; //INDPND 通过置1清零 if(count%2 == 0) rGPBDAT = 0x1; else rGPBDAT = 0xfe; count++; if(count == 20){ count = 0; rGPBDAT = 0; rTICNT &= "(1<<7); //中断20次,然后停止节拍时间中断 } } int Main(){ flag = 0; count = 0; t = 0; rGPFCON &= 0xffcc; //0 rGPFCON |= (1<<1)|(1<<5); rGPFUP = 0xfe; rGPBCON &= 0xfffc; rGPBCON |= 0x1; rGPBUP = 0xfe; rULCON0 = 0xfff00; rULCON0 |= 0x3; rUCON0 = 0x0800; rUCON0 |= 0x05; rUBRDIV0 = 26; rRTCALM = (1<<1)|(1<<6); //主要全局报警使能,然后是分钟报警使能 rSUBSRCPND |= 0x3; rSRCPND |= (1<<0)|(1<<2)|(1<<8)|(1<<28)|(1<<30); //先清一下 rINTPND |= (1<<0)|(1<<2)|(1<<8)|(1<<28)|(1<<30); //先清一下 rINTMSK &= ("(0x1<<0))&("(0x1<<2))&("(0x1<<8))&("(0x1<<28))&("(0x1<<30)); //开中断 rINTSUBMSK &= ("(0x1<<0)) & ("(0x1<<1)); pISR_EINT0 = (U32)EINT0_ISR; pISR_EINT2 = (U32)EINT2_ISR; pISR_TICK = (U32)Tick_ISR; pISR_RTC = (U32)Alarm_ISR; pISR_UART0 = (U32)Uart_ISR; Init_LCD(); Paint_background(0xffffff,0,0,320,240); SetAlarm(); SetTime(); ShowTime(); while(1){ if(flag){ Paint_background(0xffffff,0,0,320,240); ShowTime(); flag = 0; } } } 李万鹏 |
网友评论