EDN首页   博客首页 用户登陆  |  注册
aaa
发表于 2007/6/19 10:18:46

2

关于投票

avr利用pwm控制led光暗及峰鳴器音量大小

 

//ICC-AVR application builder // 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

 

 

TCNT0的初始值的计算和设置,

根据myhk007 提供的算法

最原始的方法:

T0记到255溢出为例,如果AVR的主频是8MT01024分频的话,那么T0每加1,需要的时间就是1024/8000000,加255次的总共延时就是1024/8000000*255,依次类推。

系统分类: 单片机  |  用户分类: AVR单片机学习  |  标签: 无标签  |  来源: 转贴  | 

点击查看原文

发表评论 阅读全文(1167) | 回复(0)

发表于 2007/6/19 9:10:59

2

关于投票

AVR的PWM波(转)

(1)一个实例:

这个程序是用ICC的向导生成的,很简单。

T0是作为普通8位定时器,频率100KHz,每次中断将PB0pin1)状态反转,产生的是200KHz占空比50%的方波。

T1是作为工作模式9:相频可调PWM波发生器,频率初始化16KHz,占空比50%。请注意:

TCNT1T0的定时器计数值,就是每个定时器时钟加1,和普通定时器的计数值寄存器作用一样。

OCR1A作为比较的TOP值。 OCR1B作为匹配输出值。

TCNT1的值增加到OCR1B相等时,OC1B(pin18)清零,就是对应低电平;

然后TCNT1继续增加到OCR1A(就是TOP)的值,然后TCNT1开始减少,这个中间,OC1BPin18)状态不变;当TCNT1减少到OCR1B相等时,OC1Bpin18)置1,就是对应高电平。 然后TCNT1继续减少到0x00(就是BOTTOM),然后TCNT1又开始增加,这个中间,OC1Bpin18)状态不变。

OCR1B的值与OCR1A的比值就是PWM的占空比! 所以这个值必须比OCR1A小。当OCR1B0时,PWM波就一直为低电平(相当于占空比为0);当OCR1BOCR1A时,PWM波就一直为高电平(相当于占空比为100);当OCR1BOCR1A的一半时,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%的方波,高电平上平分为12两段,低电平上平分为34两段。

1就是TCCNT1从初值加,-->0xFFFF阶段,这个阶段OCR1B为高电平;

2就是TCCNT10x00-->OCR1B阶段,这个阶段为高电平;匹配后,变为低电平

3就是TCCNT1OCR1B-->OCR1A阶段,这个阶段为低电平;

4就是TCCNT1OCR1A-->OCR1B阶段,这个阶段为低电平;匹配后,变为高电平

TCCNT1的初值,就是保证第一段高电平的时间,这样才能形成一个完整周期的方波。而且,这个初值应该根据OCR1B的值而设,就是TCCNT1 = 0xffff-OCR1B+1;这样才能保证时间的匹配。

如果是模式9,那么每次变化后,算出占空比,算出OCR1B的值并赋值,会自动在下一个周期改变占空比为新值。easy。。。重点是:每次给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 值可以为固定值0x00FF0x01FF 0x03FF,或是存储于寄存器 OCR1AICR1里的数值,具体有赖于工作模式 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=MAXTOP时也会产生TOV1溢出中断标志 

     :WGM=15时,也能实现从OC1A输出方波,而且具备双缓冲功能 

     计算公式: fOCn="fclk"_IO/(2*N*(1+TOP)) 

                   变量N 代表预分频因子(18642561024)T2多了(32128)两级。      

      

  3 快速PWM模式 WGM1=5,6,7,14,15  

    单斜波计数,用于输出高频率的PWM信号(比双斜波的高一倍频率

    都有TOV1溢出中断,发生于TOP[不是MAX,跟普通模式,CTC模式不一样

    比较匹配后可以产生OCF1x比较匹配中断

      WGM1=5, 最大值为0x00FF 8位分辨率 

      WGM1=6, 最大值为0x01FF 9位分辨率 

      WGM1=7, 最大值为0x03FF10位分辨率  

     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, 最大值为0x03FF10位分辨率  

     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=89  

    双斜波计数,用于输出高精度的、相位与频率都准确的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) 

 

系统分类: 单片机  |  用户分类: AVR单片机学习  |  标签: AVR TWM  |  来源: 转贴  | 

点击查看原文

发表评论 阅读全文(2045) | 回复(0)

发表于 2007/6/18 1:11:50

2

关于投票

软件产生PWM控制直流电机实验之 C 程序

//------------- 软件产生PWM控制直流电机实验之 C 程序----------------------
//------------------------------------------------------------------------
//公司网址: www.jingze.cn
//作者: 机器人创作室
//软件版本: ICCAVR ver 7.05
// Target : M48
// Crystal: 8.0000Mhz

#include <iom48v.h>
#include <macros.h>
unsigned int Pwmw;
unsigned int adc_v;

//ADC 转换函数
unsigned int adc_get(void)
{

ADCSRA |= (1<<ADSC); //启动ADC转换
while( ADCSRA & (1<<ADSC)); //判断ADC转换是否结束
return ADC; //取ADC的转换结果
}

//电机驱动、调速子程序
void PWMsub(unsigned int Pwmh)
{

unsigned int i;
unsigned int Pwml;
Pwml = Pwmw - Pwmh;
PORTD |= 0x10; //高电平
for (i = 0;i < Pwmh;i++)
;
PORTD = 0x00; //低电平
for (i = 0;i < Pwml;i++)
;
}
//主函数
void main()
{
PORTD = 0x00; //PD端口置0
DDRD = 0xF0; //PD7,、PD6、PD5、PD4输出
ADCSRA = 0x00; //设置前,先关闭ADC
ADMUX = (1<<REFS0) ; //选择内部AVCC为基准
ACSR = (1<<ACD); //关闭模拟比较器
//使能ADC,启动ADC单次转换,选64分频
ADCSRA = (1<<ADEN)|(1<<ADSC)|(1<<ADPS2)|(1<<ADPS1);
Pwmw = 0x03ff;
while(1)
{
adc_v = adc_get(); //ADC转换
PWMsub(adc_v);
}
}

系统分类: 单片机  |  用户分类: AVR单片机学习  |  标签: 无标签  |  来源: 转贴  | 

点击查看原文

发表评论 阅读全文(2332) | 回复(2)

发表于 2007/6/18 1:07:41

2

关于投票

60秒定时器实验

//----------------60秒定时器实验之 C 程序--------------
//------------------------------------------------------------------------
//
发行公司: 北京竟择信息技术有限公司
//
作者: 机器人创作室
//Email: robot@jingze.cn
//
软件版本: ICCAVR ver 7.05
// Target : M48
// Crystal: 8.0000Mhz

#include <iom48v.h>
#include <macros.h>
#pragma interrupt_handler timer0_handler: iv_TIMER0_OVF
#pragma interrupt_handler timer1_handler: iv_TIMER1_OVF

//call this routine to initialize all peripherals
void init_devices(void)
{
//stop errant interrupts until set up
CLI(); //disable all interrupts
PORTB = 0x00;
DDRB = 0xFF;
PORTC = 0x00;
DDRC = 0x03;
//Timer0
TCCR0B = 0x00; //stop
TCNT0 = 0x06; //set count
TCCR0A = 0x00;
TCCR0B = 0x04; //start timer
//Timer1
TCCR1B = 0x00; //stop
TCNT1H = 0x85; //setup
TCNT1L = 0xEE;
TCCR1A = 0x00;
TCCR1B = 0x04; //start Timer

TIMSK0 = 0x01; //timer 0 interrupt sources
TIMSK1 = 0x01; //timer 1 interrupt sources
SEI(); //re-enable interrupts
}

//定义字型数组
#pragma data:code
const unsigned char tabs[]=
{
0xC0,//"0"
0xF9,//"1"
0xA4,//"2"
0xB0,//"3"
0x99,//"4"
0x92,//"5"
0x82,//"6"
0xF8,//"7"
0x80,//"8"
0x98,//"9"
};

#pragma data:data
static unsigned char ledbuf[2];
unsigned char _led;
unsigned char _change;
unsigned char _second;

void main(void)
{
unsigned char digits = 0;

init_devices();
_second = 0;
while (1)
{
if (_change > 0)
{
_change = 0; //
更新显示
digits = _second / 10; //
取十位数
ledbuf[0] = tabs[digits];
digits = _second % 10; //
取个位数
ledbuf[1] = tabs[digits];
}
}
}

void timer0_handler()
{
TCNT0 = 0x06; //set count
PORTB = ledbuf[_led]; //
显示字型送PortB
if (_led == 0)
{
PORTC = 1; //PortC
控制显示
_led = 1;
}
else
{
PORTC = 2; //PortC
控制显示
_led = 0;
}
}

void timer1_handler()
{
TCNT1H = 0x85; //reload counter high value
TCNT1L = 0xEE; //reload counter low value
_change = 1; //
定时变化标记
if (_second < 59)
_second += 1;
else
_second = 0;

 

系统分类: 单片机  |  用户分类: AVR单片机学习  |  标签: 无标签  |  来源: 转贴  | 

点击查看原文

发表评论 阅读全文(715) | 回复(0)

发表于 2007/6/18 0:54:50

2

关于投票

AVR定时器的要点

(大部分摘自 M16中文手册,未能一一测试)

  M16的T1 16位定时器一共有15种工作模式,其他2个8位定时器(T0/T2)相对简单,除了T2有异步工作模式用于RTC应用外
                       (可以利用溢出中断和比较匹配中断作定时功能)
  分5种工作类型
  1  普通模式 WGM1=0
    跟51的普通模式差不多,有TOV1溢出中断,发生于TOP时
    1 采用内部计数时钟     用于 ICP捕捉输入场合---测量脉宽/红外解码
        (捕捉输入功能可以工作在多种模式下,而不单单只是普通模式)
    2 采用外部计数脉冲输入  用于 计数,测频
    其他的应用,采用其他模式更为方便,不需要像51般费神
    
  2 CTC模式 [比较匹配时清零定时器模式] WGM1=4,12
     跟51的自动重载模式差不多
     1 用于输出50%占空比的方波信号
     2 用于产生准确的连续定时信号
     WGM1=4时, 最大值由OCR1A设定,TOP时产生OCF1A比较匹配中断
     WGM1=12时,最大值由ICF1设定, TOP时产生ICF1输入捕捉中断
     注:WGM=15时,也能实现从OC1A输出方波,而且具备双缓冲功能
     计算公式: fOCn="fclk"_IO/(2*N*(1+TOP))
                   变量N 代表预分频因子(1、8、32,64、256,1024)。     
     
  3 快速PWM模式 WGM1=5,6,7,14,15 
    单斜波计数,用于输出高频率的PWM信号(比双斜波的高一倍频率)
    都有TOV1溢出中断,发生于TOP时
    比较匹配后可以产生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)

T/C 的时钟源
   T/C 的时钟源可以有多种选择,由CS12:0控制,分别用于高速(低分频)/长时间(高分频)/外部计数场合
   一个16位定时器,在8MHz系统时钟驱动下,可以实现uS级的高速定时和长达8秒的超长定时,这可是标准51的弱点

   CS12 CS11 CS10 说明
       0    0    0  无时钟源     (T/C 停止)
       0    0    1  clkIO/1    ( 无预分频) 
       0    1    0  clkIO/8    ( 来自预分频器)
       0    1    1  clkIO/64   ( 来自预分频器)
       1    0    0  clkIO/256  ( 来自预分频器) 
       1    0    1  clkIO/1024 ( 来自预分频器)
       1    1    0  外部T1 引脚,下降沿驱动
       1    1    1  外部T1 引脚,上升沿驱动
  分频器复位
        在高预分频应用时,通过复位预分频器来同步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 状态;
   2 COM1x1:0 还控制OC1x 引脚输出的来源。
   只要COM1x1:0 不全为零,波形发生器的输出比较功能就会重载OC1x 的通用I/O 口功能。
   但是OC1x 引脚的方向仍旧受控于数据方向寄存器 (DDR)。
   从OC1x 引脚输出有效信号之前必须通过数据方向寄存器的DDR_OC1x 将此引脚设置为输出。
   
   波形发生器利用COM1x1:0 的方法在普通模式、CTC 模式和PWM 模式下有所区别。
   对于所有的模式,设置COM1x1:0=0 表明比较匹配发生时波形发生器不会操作OC1x寄存器

访问16位寄存器
   写16 位寄存器时,应先写入该寄存器的高位字节.
      usigned int k;
      k="0x1234";
      TCNT1H=(unsigned char)(k>>8);
      TCNT1L=(unsigned char) k;
   而读16 位寄存器时应先读取该寄存器的低位字节.
      usigned int k;
       k="TCNT1L";
       k+=(unsigned int)(TCNT1H<<8);
   使用“C” 语言时,编译器会自动处理16位操作.
      usigned int k;
      k="0x1234";
      TCNT="k";
      k="TCNT1";

系统分类: 单片机  |  用户分类: AVR单片机学习  |  标签: avr 定时器  |  来源: 转贴  | 

点击查看原文

发表评论 阅读全文(1260) | 回复(0)

发表于 2007/6/16 21:07:27

2

关于投票

SL isp 使用小记

       今天晚上换了台电脑写程序,生成HEX文件以后,在烧写片子的时候遇到个问题,跳出个对话框,提示找不到并口,我以为是我设置有问题,就翻来覆去的在软件上找毛病,半个小时没解决,我就有换了台电脑,还是不行,把我闷那里了。

       突然我想到用51的ISP写程序时,需要注意必须把CMOS中的并口设置成ECP模式。是不是这里出问题了,因为这条下载线我刚用过,肯定是好的。刚才我又把线拆开看看,没问题,肯定是电脑的问题。

       我重启了计算机,在CMOS中找到并口设置项,改成ECP模式,再启动,成功了,还是这个问题。

       解决了,写下来,给同学AVR的伙伴们共勉。

系统分类: 单片机  |  用户分类: AVR单片机学习  |  标签: 无标签  |  来源: 原创  | 

点击查看原文

发表评论 阅读全文(1135) | 回复(0)

发表于 2007/6/15 9:35:54

2

关于投票

ATmega16的熔丝设置详解

初学者对熔丝经常不解,AVR芯片使用熔丝来设定时钟、启动时间、一些功能的使能、BOOT区设定、当然还有最让初学者头疼的保密位,设不好锁了芯片很麻烦。要想使MCU功耗最小也要了解一些位的设定

在此写下自己对熔丝的理解,参照了一些MEGA16PDF文档,双龙的文档,以及大家的帖子。力求易懂、全面。

下面以双龙的在系统编程软件SLISP为例具体说明我对熔丝的理解。

a intro

b 低位(时钟及启动时间设置):

c 时钟总表

d 位(BOOT区设置)

e 常用熔丝设置

 打开运行SLISP.exe,首先记住:

1:未编程(配置熔丝检查框未打钩

0:编程 (配置熔丝检查框打钩

建议在配置熔丝之前先“读取配置”读出原来的设定,再自己编辑。

先了解一下M16的出厂设置。

默认设置为:内部RC振荡8MHz 6 CK + 65 ms CKSEL="0100" SUT="10"低位(时钟及启动时间设置):

1.BOD(Brown-out Detection) 掉电检测电路

BODLEVEL(BOD电平选择): 1: 2.7V电平; 04.0V电平

BODEN(BOD功能控制): 1BOD功能禁止;0BOD功能允许

使用方法:如果BODEN使能(复选框选中)启动掉电检测,则检测电平由BODLEVEL决定。一旦VCC下降到触发电平(2.7v4.0v)以下,MCU复位;当VCC电平大于触发电平后,经过tTOUT 延时周后重新开始工作。

注:1.复选框选中代表00电平有效。            

   2.因为M16L可以工作在2.7v5.5v,所以触发电平可选2.7v(BODLEVEL=0)4.0v(BODLEVEL=1);M16工作在4.55.5V,所以只能选BODLEVEL=0BODLEVEL=1不适用于ATmega16

2.复位启动时间选择

SUT 1/0:  当选择不同晶振时,SUT有所不同。

如果没有特殊要求推荐SUT 1/0设置复位启动时间稍长,使电源缓慢上升。

CKSEL3/0: 时钟源选择(时钟总表

 

时钟总表 

  时钟源               启动延时         熔丝

  外部时钟             6 CK + 0 ms               CKSEL=0000 SUT="00"

  外部时钟             6 CK + 4.1 ms             CKSEL=0000 SUT="01"

  外部时钟             6 CK + 65 ms              CKSEL="0000" SUT="10"

内部RC振荡1MHZ                    6 CK + 0 ms    CKSEL="0001" SUT="00"

内部RC振荡1MHZ