对于很多电子爱好者来说,自己制作一款LED的电子钟,是很有意思的一件事情,LED的电子钟虽然耗电一点,但在夜间也不用打开照明就可以看得清清楚楚,还是很方便的了.
PIC16C54内部有512字节(准确的说是可以放512条指令)的指令空间,对于电子钟的应用项目来说,资源已经足够了.它具有12个I/O口,刚好可以用来做一个四个数码管的电子钟.其中PORTA口用作位码输出,PORTB用作段码输出和按键输入.
54内部有一个8位的定时器,但没有中断溢出功能,对于用惯了中断的人来说,可能觉得定时基准不好做,比如51系列的,只要设定好中断溢出时间,一般取整数指令周期,如每次溢出时间为50mS,每次溢出时累加器加一,当加到20次时就有1S了,很方便进行系统时钟的处理.但PIC16C54没有中断功能,只能用判断定时器溢出的方式来确定定时器的定时.判断定量器溢出可以采用比较的方式,当采集到TMR0的计数值为0-5时,可以认为定时器溢出了,另一种方式是测试TMR0的最高位是否为1,也就是把TMR0当作7位定时器来用,这样,就不会出现前一种方式由于程序错过捕获到TMR0的值为0-5的时机而使定时出现误差了.比较好的方法是用一个存储器X0跟踪TMR0的计数值,在正常情况下,X0总是会小于TMR0的计数值,因为在读取TMR0的值并把这个值赋值给X0之后,TMR0的计数值又在累加了,但TMR0是一个循环计数器,当加到255后,其值将会变为零,这样就为判断TMR0提供了依据,即只要测试到X0>=TMR0,即可认为TMR0已经溢出,这样就可以进行相应的处理了.如时钟频率为4MHz,则一条指令的执行时间为1uS,TMR0溢出对应于256uS,当测试到TMR0溢出之后,不对TMR0进行任何赋值操作,因为对TMR0进行写入,会使TMR0延迟三个指令周期,而每次往TMR0写入的时候,不可能它的计数值都是同一个值,所以只能采用加一个预定值的方法,如果TMR0不采用分频器,则对TMR0执行 TMR0=TMR0+9的操作,将使每次TMR0的计数溢出周期等效为250个指令周期,也就是250uS,需要注意的是,每次读入TMR0的值与X0进行比较时的时间间隔一定不能大于TMR0的溢出时间,否则错过溢出时刻.判断TMR0溢出的最短指令为:
MOVF TMR0,W
SUBWF X0,F
MOVWF X0
SKPC
GOTO 没有溢出...
溢出处理...
对应的C语言语句为:
i=TMR0;
x0=x0-i;
x0=i;
if(CARRY)
{TMR0+=9;
}
这里借用了一个临时存储器i,如果没有它,则语句编译不会最短,程序执行会了出错,不能保证每次溢出为250uS.
如果选用32768晶振的话,程序的设计上,不需要再对TMR0作任何写入,只需用上述程序判断TMR0溢出就可以了.因为TMR0每次溢出的时间为0.03125秒,再对其进行32分频即可得到秒信号了.
这样,整个程序的核心部分就完成了,剩下的应该不难写出来.