s3c2440内部集成了RTC模块,而且用起来也十分简单。其内部的寄存器BCDSEC,BCDMIN,BCDHOUR,BCDDAY,BCDDATE,BCDMON和BCDYEAR分别存储了当
前的秒,分,小时,星期,日,月和年,表示时间的数值都是BCD码。这些寄存器的内容可读可写,并且只有在寄存器RTCCON的第0位为1时才能进行读写操作。为了防止
误操作,当不进行读写时,要把该位清零。当读取这些寄存器时,能够获知当前的时间;当写入这些寄存器时,能够改变当前的时间。另外需要注意的是,因为有所谓的“一秒
误差”,因此当读取到的秒为0时,需要重新再读取一遍这些寄存器的内容,才能保证时间的正确(关于这点可以看s3c2440文档)。
下面给出一个简单的程序,当刚上电的时候读取当前时间值,通过串口显示出来,并设置当前的时间为 2014年8月25日,21点40分 33秒:
设置完毕之后注释掉设置时间的代码再次编译 ,通过openjtag down进去,可见到在关闭电源的一段时间内时钟也在计时。这算是个简单的 小功能,可以在我github
上clone。
除了简单的计时之外RTC还提供了2种时钟中断功能:报警中断和时间节拍中断,时间节拍中断,顾名思义,就像一个节拍器,可以等时性的控制节拍。因此它类
似于定时器中断。但时间节拍中断是毫秒级的,而定时器中断可以达到微秒,甚至更小级别。时间节拍中断的周期公式为:(n+1)÷128,单位是秒,即每隔这
么长时间,会中断一次。其中n的值为1~127,它存储在寄存器TICNT的低6位中,当寄存器TICNT的第7位被置1时,表示开启时间节拍中断,这时n递减,当减
为0时,进入时间节拍中断。一般在ucos里面的节拍就是ms级别,作为ucos的节拍比较合适。
报警中断可以实现当实时时间达到预置的时间后,引起报警。预置的时间是存储在报警时间数据寄存器中的(rtc有2套时间有关的寄存器,都是BCD形式存储。当前时间用
一套,),包括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的相应位来比较相对应的当前时间与报警时间数据寄存器中的值,如果相等就进入中断,不论设置什么形式的闹钟中断全局使能需要置位。
可以根据这个实现一些功能:时间节拍中断设置为1s一次,每次节拍中断向串口写一串字符"beat int"。而刚上电之后设置闹钟中断则是在上电之后1分钟之后向串口写字符
串"alarm int"。
原文地址:http://blog.csdn.net/xsckernel/article/details/38826875