1:AVR 定时/计数器的PWM
模式可以分成快速PWM和频率(相位)调整PWM 两大类。
快速PWM 可以得到比较高频率的PWM 输出,但占空比的调节精度稍微差一些。此时计数器仅工作在单程
正向计数方式,计数器的上限值决定PWM的频率,而比较匹配寄存器的值决定了占空比的大小。PWM 频率
的计算公式为:
PWM频率 = 系统时钟频率/(分频系数*(1+计数器上限值))
频率(相位)调整PWM 模式的占空比调节精度高,但输出频率比较低,因为此时计数器仅工作在双向计数方式。同样计数器的上限值决定了PWM 的频率,比较匹配寄存器的值决定了占空比的大小。PWM 频率的计
算公式为:
PWM频率 = 系统时钟频率/(分频系数*2*计数器上限值))
在PWM 方式中,计数器的上限值有固定的0xFF(8 位T/C);0xFF、0x1FF、0x3FF(16 位T/C)。或由用户设定的0x0000-0xFFFF,设定值在16 位T/C 的ICP 或OCRA 寄存器中。而比较匹配寄存器的值与计数器上限值之比即为占空比。
2:应用例子
用PWM 方式来产生一个1KHz 左右的正弦波,幅度为0-Vcc/2。
程序中使用ATmega128 的8 位T/C0,工作模式为相位调整PWM 模式输出,系统时钟为8MHz,分频系
数为1,其可以产生最高PWM 频率为: 8000000Hz / 510 = 15686Hz。每16 次输出构成一个周期正弦波,正弦波的频率为980.4Hz。PWM 由OC0(PB4)引脚输出。参考程序如下(ICCAVR)。
//ICC-AVR application builder : 2009-04
// Target : M128
// Crystal: 8.0000Mhz
#include <iom128v.h>
#include <macros.h>
#pragma data:code //随后的数据存放在Flash区域
// 128 点正弦波样本表
const unsigned char auc_SinParam[128] = {
64,67,70,73,76,79,82,85,88,91,94,96,99,102,104,106,109,111,113,115,117,118,120,121,
123,124,125,126,126,127,127,127,127,127,127,127,126,126,125,124,123,121,120,118,
117,115,113,111,109,106,104,102,99,96,94,91,88,85,82,79,76,73,70,67,64,60,57,54,51,48,
45,42,39,36,33,31,28,25,23,21,18,16,14,12,10,9,7,6,4,3,2,1,1,0,0,0,0,0,0,0,1,1,2,3,4,6,
7,9,10,12,14,16,18,21,23,25,28,31,33,36,39,42,45,48,51,54,57,60};
#pragma data:data//随后数据放在RAM区
unsigned char x_SW = 8,X_LUT = 0;
#pragma interrupt_handler timer0_ovf_isr:17//中断常量表位置
void timer0_ovf_isr(void)
{
X_LUT += x_SW; // 新样点指针//学习此种编程风格
if (X_LUT > 127)
X_LUT -= 128; // 样点指针调整
OCR0 = auc_SinParam[X_LUT]; // 取样点指针到比较匹配寄存器
}
void main(void)
{
DDRB |= 0x10; // PB4(OC0)输出
TCCR0 = 0x71; // 相位调整PWM 模式,分频系数=1,正向控制OC0
/*TCCR0
• Bit 7 – FOC0: 强制输出比较
FOC0仅在WGM指明非PWM模式时才有效。但是,为了保证与未来器件的兼容性,在使
用PWM 时,写TCCR0 要对其清零。对其写1 后,波形发生器将立即进行比较操作。比
较匹配输出引脚 OC0 将按照COM01:0 的设置输出相应的电平。要注意FOC0 仅仅是一
个启动信号,真正对强制输出比较起作用的是COM01:0 的设置。
OCFn
OCRn
TCNTn
OCRn Value
OCRn - 1 OCRn OCRn + 1 OCRn + 2
clkI/O
clkTn
(clkI/O/8)
OCFn
OCRn
TCNTn
(CTC)
TOP
TOP - 1 TOP BOTTOM BOTTOM + 1
clkI/O
clkTn
(clkI/O/8)
Bit 7 6 5 4 3 2 1 0
FOC0 WGM00 COM01 COM00 WGM01 CS02 CS01 CS00 TCCR0
读/ 写W R/W R/W R/W R/W R/W R/W R/W
初始值0 0 0 0
Cs2,Cs1,Cs0 =1,预分频系数为1
WGM01 WGM00=1 相位校正PWM模式
*/
TIMSK = 0x01; // T/C0 溢出中断允许
SEI(); // 使能全局中断
while(1)
{……};
}
每次计数器溢出中断的服务中取出一个正弦波的样点值到比较匹配寄存器中,用于调整下一个PWM 的脉
冲宽度,这样在PB4 引脚上输出了按正弦波调制的PWM 方波。当PB4 的输出通过一个低通滤波器后,便得
到一个980.4Hz 的正弦波了。如要得到更精确的1KHz 的正弦波,可使用定时/计数器T/C1,选择工作模式
10,设置ICR1=250 为计数器的上限值
AVR的PWM波
(1)一个实例:
这个程序是用ICC的向导生成的,很简单。
T0是作为普通8位定时器,频率100KHz,每次中断将PB0(pin1)状态反转,产生的是200KHz占空比50%的方波。
T1是作为工作模式9:相频可调PWM波发生器,频率初始化16KHz,占空比50%。请注意:
TCNT1是T0的定时器计数值,就是每个定时器时钟加1,和普通定时器的计数值寄存器作用一样。
OCR1A作为比较的TOP值。 OCR1B作为匹配输出值。
当TCNT1的值增加到OCR1B相等时,OC1B(pin18)清零,就是对应低电平;
然后TCNT1继续增加到OCR1A(就是TOP)的值,然后TCNT1开始减少,这个中间,OC1B(Pin18)状态不变;当TCNT1减少到OCR1B相等时,OC1B(pin18)置1,就是对应高电平。 然后TCNT1继续减少到0x00(就是BOTTOM),然后TCNT1又开始增加,这个中间,OC1B(pin18)状态不变。
OCR1B的值与OCR1A的比值就是PWM的占空比!所以这个值必须比OCR1A小。当OCR1B为0时,PWM波就一直为低电平(相当于占空比为0);当OCR1B为OCR1A时,PWM波就一直为高电平(相当于占空比为100);当OCR1B为OCR1A的一半时,PWM波就是占空比为50%。
你可以修改OCR1B的值,然后重新下载程序运行,看看占空比的改变;也可以修改OCR1A的值,然后重新下载程序运行,看看频率的改变,不过要注意修改OCR1A时,同时注意OCR1B的值不要比OCR1A大。
模式9算是PWM生成中最复杂的一种,只要你理解了这个,对别的几种PWM都好理解。
TCNT0 = 0xB0; //set count
OCR0 = 0x50;
即使工作在normal模式下,这个OCR0仍然在和TCNT0进行比较,一旦匹配后,就会产生中断或者改变OC0脚上的电平(产生PWM)。改变这个值,就会改变中断发生的时间,或者改变OC0脚上的方波的频率了。
T1定时器1的模式9,相频修正模式,可以用来产生波形非常完整的PWM波。TCNT1设置初值,增加到0xFFFF的时间,然后从0开始计数,这个理解是正确的。可以画一个波形图对应理解一下:画一个占空比50%的方波,高电平上平分为1、2两段,低电平上平分为3、4两段。
1就是TCCNT1从初值加,-->0xFFFF阶段,这个阶段OCR1B为高电平;
2就是TCCNT1从0x00加-->OCR1B阶段,这个阶段为高电平;匹配后,变为低电平
3就是TCCNT1从OCR1B加-->OCR1A阶段,这个阶段为低电平;
4就是TCCNT1从OCR1A减-->OCR1B阶段,这个阶段为低电平;匹配后,变为高电平
TCCNT1的初值,就是保证第一段高电平的时间,这样才能形成一个完整周期的方波。而且,这个初值应该根据OCR1B的值而设,就是TCCNT1 = 0xffff-OCR1B+1;这样才能保证时间的匹配。
如果是模式9,那么每次变化后,算出占空比,算出OCR1B的值并赋值,会自动在下一个周期改变占空比为新值。
重点是:每次给OCR1B赋值,会在 下一个 周期改变占空比。
//实例:利用pwm控制led光暗及峰鳴器音量大小
//ICC-AVR application builder : 2005-4-18 12:46:03
// Target : M16
// Crystal: 4.0000Mhz
#include <iom16v.h>
#include <macros.h>
#define uchar unsigned char
#define uint unsigned int
void port_init(void);
void timer0_init(void);
void init_devices(void);
void delay_short(uint t);
uchar scan_key(void);
void port_init(void)
{
PORTA = 0x00;
DDRA = 0x00;
PORTB = BIT(PB3);
DDRB = BIT(PB3);
PORTC = 0x00; //m103 output only
DDRC = 0x00;
PORTD = 0x00;
DDRD = 0x00;
}
// WGM: PWM Phase correct
// desired value: 1KHz
// actual value: 0.980KHz (-2.0%)
void timer0_init(void)
{
TCCR0 = 0x00; //stop
TCNT0 = 0x01; //set count
OCR0 = 0xFF; //set compare
TCCR0 = 0x62; //start timer ; 相位修正, 8分頻
}
//call this routine to initialize all peripherals
void init_devices(void)
{
//stop errant interrupts until set up
CLI(); //disable all interrupts
port_init();
timer0_init();
MCUCR = 0x00;
GICR = 0x00;
TIMSK = 0x00; //timer interrupt sources
SEI(); //re-enable interrupts
//all peripherals are now initialized
}
void delay_short(uint t) // 短延時
{
uint i;
for (i=0;i<t;i++);
}
uchar scan_key(void) // 按鍵掃瞄
{
uchar v;
v = 0;
if ((PIND & 0x07) != 0x07)
{
if ((PIND & 0x01) == 0)
{
v = 1;
delay_short(1000);
}
if ((PIND & 0x2) == 0)
{
v = 2;
delay_short(1000);
}
if ((PIND & 0x4) == 0)
{
v = 3;
delay_short(1000);
}
};
while((PIND & 0x07) != 0x07); // 判斷按鍵是不是放開
return v;
}
//
void main(void)
{
uchar key, OCR0_V;
init_devices();
OCR0_V = 0xff;
while(1)
{
key = scan_key();
if (key > 0)
{
if (key==1) // 減少佔空比
{
OCR0_V -= 10;
OCR0 = OCR0_V;
};
if (key==2) // 增加佔空比
{
OCR0_V += 10;
OCR0 = OCR0_V;
};
if (key==3) // 全黑,佔空比為100%
{
OCR0_V = 0xff;
OCR0 = OCR0_V;
};
}
};
}
實驗板接線:
PB3 -----> JA.1 及 JM
PD0 -----> K1
PD1 -----> K2
PD2 -----> K3
(2)相关详细理论说明:
符号定义:
BOTTOM 计数器计到0x0000 时即达到BOTTOM
MAX 计数器计到0xFFFF ( 十进制的65535) 时即达到MAX
TOP 计数器计到计数序列的最大值时即达到TOP。
TOP 值可以为固定值0x00FF、0x01FF或 0x03FF,或是存储于寄存器 OCR1A或ICR1里的数值,具体有赖于工作模式 分5种工作类型
1 普通模式 WGM1=0
跟51的普通模式差不多,有TOV1溢出中断标志,发生于MAX(0xFFFF)时
1 采用内部计数时钟 用于 ICP捕捉输入场合---测量脉宽/红外解码
(捕捉输入功能可以工作在多种模式下,而不单单只是普通模式)
2 采用外部计数脉冲输入 用于 计数,测频
其他的应用,采用其他模式更为方便,不需要像51般费神
2 CTC模式 [比较匹配时清零定时器模式] 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)