M16的T116位定时器一共有15种工作模式,其他2个8位定时器(T0/T2)相对简单,除了T2有异步工作模式用于RTC应用外(可以利用溢出中断和比较匹配中断作定时功能)
符号定义:
BOTTOM计数器计到0x0000时即达到BOTTOM
MAX计数器计到0xFFFF(十进制的65535)时即达到MAX
TOP计数器计到计数序列的最大值时即达到TOP。
TOP值可以为固定值0x00FF、0x01FF或0x03FF,或是存储于寄存器OCR1A或ICR1里的数值,具体有赖于工作模式
------注意MAX和TOP是不同的,在表格[波形产生模式的位描述]可以看到它们的作用
分5种工作类型
1普通模式WGM1=0
跟51的普通模式差不多,有TOV1溢出中断标志,发生于MAX(0xFFFF)时
1采用内部计数时钟用于ICP捕捉输入场合---测量脉宽/红外解码
(捕捉输入功能可以工作在多种模式下,而不单单只是普通模式)
2采用外部计数脉冲输入用于计数,测频
其他的应用,采用其他模式更为方便,不需要像51般费神
2CTC模式[比较匹配时清零定时器模式]WGM1=4,12
跟51的自动重载模式差不多
1用于输出50%占空比的方波信号
2用于产生准确的连续定时信号
WGM1=4时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断标志
WGM1=12时,最大值由ICF1设定,TOP时产生ICF1输入捕捉中断标志
------如果TOP=MAX,TOP时也会产生TOV1溢出中断标志
注:WGM=15时,也能实现从OC1A输出方波,而且具备双缓冲功能
计算公式:fOCn=fclk_IO/(2*N*(1+TOP))
变量N代表预分频因子(1、8、64、256、1024),T2多了(32、128)两级。
3快速PWM模式WGM1=5,6,7,14,15
单斜波计数,用于输出高频率的PWM信号(比双斜波的高一倍频率)
都有TOV1溢出中断,发生于TOP时[不是MAX,跟普通模式,CTC模式不一样]
比较匹配后可以产生OCF1x比较匹配中断.
WGM1=5时,最大值为0x00FF,8位分辨率
WGM1=6时,最大值为0x01FF,9位分辨率
WGM1=7时,最大值为0x03FF,10位分辨率
WGM1=14时,最大值由ICF1设定,TOP时产生ICF1输入捕捉中断(单缓冲)
WGM1=15时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断(双缓冲,但OC1A将没有PWM能力,最多只能输出方波)
改变TOP值时必须保证新的TOP值不小于所有比较寄存器的数值
注意,即使OCR1A/B设为0x0000,也会输出一个定时器时钟周期的窄脉冲,而不是一直为低电平
计算公式:fPWM=fclk_IO/(N*(1+TOP))
4相位修正PWM模式WGM1=1,2,3,10,11
双斜波计数,用于输出高精度的,相位准确的,对称的PWM信号
都有TOV1溢出中断,但发生在BOOTOM时
比较匹配后可以产生OCF1x比较匹配中断.
WGM1=1时,最大值为0x00FF,8位分辨率
WGM1=2时,最大值为0x01FF,9位分辨率
WGM1=3时,最大值为0x03FF,10位分辨率
WGM1=10时,最大值由ICF1设定,TOP时产生ICF1输入捕捉中断(单缓冲)
WGM1=11时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断(双缓冲,但OC1A将没有PWM能力,最多只能输出方波)
改变TOP值时必须保证新的TOP值不小于所有比较寄存器的数值
可以输出0%~100%占空比的PWM信号
若要在T/C运行时改变TOP值,最好用相位与频率修正模式代替相位修正模式。若TOP保持不变,那么这两种工作模式实际没有区别
计算公式:fPWM=fclk_IO/(2*N*TOP)
5相位与频率修正PWM模式WGM1=8,9
双斜波计数,用于输出高精度的、相位与频率都准确的PWM波形
都有TOV1溢出中断,但发生在BOOTOM时
比较匹配后可以产生OCF1x比较匹配中断.
WGM1=8时,最大值由ICF1设定,TOP时产生ICF1输入捕捉中断(单缓冲)
WGM1=9时,最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断(双缓冲,但OC1A将没有PWM能力,最多只能输出方波)
相频修正修正PWM模式与相位修正PWM模式的主要区别在于OCR1x寄存器的更新时间
改变TOP值时必须保证新的TOP值不小于所有比较寄存器的数值
可以输出0%~100%占空比的PWM信号
使用固定TOP值时最好使用ICR1寄存器定义TOP。这样OCR1A就可以用于在OC1A输出PWM波。
但是,如果PWM基频不断变化(通过改变TOP值),OCR1A的双缓冲特性使其更适合于这个应用。
计算公式:fPWM=fclk_IO/(2*N*TOP)
快速参考见Table47.波形产生模式的位描述
T/C的时钟源
T/C的时钟源可以有多种选择,由CS12:0控制,分别用于高速(低分频)/长时间(高分频)/外部计数场合
一个16位定时器,在8MHz系统时钟驱动下,可以实现uS级的高速定时和长达8秒的超长定时,这可是标准51的弱点
CS12CS11CS10说明
分频器复位
在高预分频应用时,通过复位预分频器来同步T/C与程序运行,可以减少误差。
但是必须注意另一个T/C是否也在使用这一预分频器,因为预分频器复位将会影响所有与其连接的T/C。
外部时钟源
由于使用了引脚同步逻辑,建议外部时钟的最高频率不要大于fclk_IO/2.5。
外部时钟源不送入预分频器
选择使用外部时钟源后,即使T1引脚被定义为输出,其T1引脚上的逻辑信号电平变化仍然会驱动T/C1计数,这个特性允许用户通过软件来控制计数。
输入捕捉单元
T/C的输入捕捉单元可用来捕获外部事件,并为其赋予时间标记以说明此时间的发生时刻。
外部事件发生的触发信号由引脚ICP1输入,也可通过模拟比较器单元来实现。
时间标记可用来计算频率、占空比及信号的其它特征,以及为事件创建日志。
输入捕捉单元可以工作在多种工作模式下
(使用ICR1定义TOP的(WGM1=12,14,10,8)波形产生模式时,ICP1与输入捕捉功能脱开,从而输入捕捉功能被禁用。)
在任何输入捕捉工作模式下都不推荐在操作过程中改变TOP值
当引脚ICP1上的逻辑电平(事件)发生了变化,或模拟比较器输出ACO电平发生了变化,并且这个电平变化为边沿检测器所证实,输入捕捉即被激发:
16位的TCNT1数据被拷贝到输入捕捉寄存器ICR1,同时输入捕捉标志位ICF1置位。
如果此时ICIE1=1,输入捕捉标志将产生输入捕捉中断。
中断执行时ICF1自动清零,或者也可通过软件在其对应的I/O位置写入逻辑"1”清零。
注意,改变触发源有可能造成一次输入捕捉。因此在改变触发源后必须对输入捕捉标志执行一次清零操作以避免出现错误的结果
除去使用ICR1定义TOP的波形产生模式外,T/C中的噪声抑制器与边沿检测器总是使能的。
(其实就是永远使能??)
使能噪声抑制器后,在边沿检测器前会加入额外的逻辑电路并引入4个系统时钟周期的延迟.
噪声抑制器使用的是系统时钟,因而不受预分频器的影响
使用输入捕捉中断时,中断程序应尽可能早的读取ICR1寄存器
如果处理器在下一次事件出现之前没有读取ICR1的数据,ICR1就会被新值覆盖,从而无法得到正确的捕捉结果。
测量外部信号的占空比时要求每次捕捉后都要改变触发沿。
因此读取ICR1后必须尽快改变敏感的信号边沿。改变边沿后,ICF1必须由软件清零(在对应的I/O位置写"1”)。
若仅需测量频率,且使用了中断发生,则不需对ICF1进行软件清零。
输出比较单元
16位比较器持续比较TCNT1与OCR1x的内容,一旦发现它们相等,比较器立即产生一个匹配信号。
然后OCF1x在下一个定时器时钟置位。
如果此时OCIE1x=1,OCF1x置位将引发输出比较中断。
(就是说输出比较可以工作在所有工作模式下,但PWM模式下更好用,功能更强)
输出比较单元A(OCR1A)的一个特质是定义T/C的TOP值(即计数器的分辨率)。
TOP值还用来定义通过波形发生器产生的波形的周期。
由于在任意模式下写TCNT1都将在下一个定时器时钟周期里阻止比较匹配,在使用输出比较时改变TCNT1就会有风险,不管T/C是否在运行
这个特性可以用来将OCR1x初始化为与TCNT1相同的数值而不触发中断。
强制输出比较(FOC)
工作于非PWM模式时,可以通过对强制输出比较位FOC1x写”1”的方式来产生比较匹配。
强制比较匹配不会置位OCF1x标志,也不会重载/清零定时器,
但是OC1x引脚将被更新,好象真的发生了比较匹配一样(COMx1:0决定OC1x是置位、清零,还是交替变化)。
比较匹配输出单元
比较匹配模式控制位COM1x1:0具有双重功能。
1波形发生器利用COM1x1:0来确定下一次比较匹配发生时的输出比较OC1x状态;
2COM1x1:0还控制OC1x引脚输出的来源。
只要COM1x1:0不全为零,波形发生器的输出比较功能就会重载OC1x的通用I/O口功能。
但是OC1x引脚的方向仍旧受控于数据方向寄存器(DDR)。
从OC1x引脚输出有效信号之前必须通过数据方向寄存器的DDR_OC1x将此引脚设置为输出。
波形发生器利用COM1x1:0的方法在普通模式、CTC模式和PWM模式下有所区别。
对于所有的模式,设置COM1x1:0=0表明比较匹配发生时波形发生器不会操作OC1x寄存器
访问16位寄存器
写16位寄存器时,应先写入该寄存器的高位字节.
usignedintk;
k=0x1234;
TCNT1H=(unsignedchar)(k>>8);
TCNT1L=(unsignedchar)k;
而读16位寄存器时应先读取该寄存器的低位字节.
usignedintk;
k=TCNT1L;
k+=(unsignedint)(TCNT1H<<8);
使用“C”语言时,编译器会自动处理16位操作.
usignedintk;
k=0x1234;
TCNT1=k;
k=TCNT1;
这里举例如何用16位定时器T1实现高精度1秒连续定时,精准度跟所用晶振一样
T1CTC模式,8MHz外部晶振,定时1秒的话,选256分频,刚好整步距,非常准确TOP=1000000/(0.125*256)-1=31249=0x7A11。
作连续定时,必用CTC/PWM模式作,没有累积误差,稳定度跟时钟是一样,手动重装受中断影响是很难达到的。
不过定时步距和最长定时间取决于时钟,分频系数和模式。
对于非整步距的定时时间要求,就会存在小于一个步距的偏差。
例如T1,CTC模式,8MHz
fOCn=fclk_IO/(2*N*(1+TOP))
定时时间T=0.125uS*N*(1+TOP)
定时1秒的话,选256分频,刚好整步距,非常准确,TOP=1000000/(0.125*256)-1=31249=0x7A11。
-----------当然了,时钟必须是高精度的晶振之类,不要用内部RC振荡器来瞎搞。
由于精度取决于晶振的精度,配合软件做RTC实时时钟是完全可行的。
用+/-20PPM的晶振,跑一个月误差1分钟[60*60*24*30=2592000秒*20ppm=52秒]。
作RTC用专门为32.768KHz时钟优化的T2定时器更合适,这里只是举例T1的实现方法
如果用+/-2.5PPM的DS32KHz(MAXIM的业界最准确的32.768KHz单片稳补时钟芯片TXCO)做时钟源,超准确
指标:-40~+85度全温度范围,年误差<4分钟,0~40度温度范围内+/-1PPM,年误差<1分钟