S3C2440-RTC中断

发布时间:2011-3-25 14:25    发布者:techshare
关键词: RTC , S3C2440 , 中断
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;  
        }  
    }  
}


李万鹏
本文地址:https://www.eechina.com/thread-59910-1-1.html     【打印本页】

本站部分文章为转载或网友发布,目的在于传递和分享信息,并不代表本网赞同其观点和对其真实性负责;文章版权归原作者及原出处所有,如涉及作品内容、版权和其它问题,我们将根据著作权人的要求,第一时间更正或删除。
您需要登录后才可以发表评论 登录 | 立即注册

厂商推荐

相关在线工具

相关视频

关于我们  -  服务条款  -  使用指南  -  站点地图  -  友情链接  -  联系我们
电子工程网 © 版权所有   京ICP备16069177号 | 京公网安备11010502021702
快速回复 返回顶部 返回列表