新闻中心

EEPW首页>嵌入式系统>设计应用> AVR单片机(学习ing)-ATMEGA16的定时/计数器

AVR单片机(学习ing)-ATMEGA16的定时/计数器

作者: 时间:2016-11-27 来源:网络 收藏
初始化:


然后是按下INT0:


最后是按下S1:


基本上就这点吧~~这里边比较难理解的就是那个去取反的重定义!!!!再IAR工具的应用里会有介绍(就是持续更新的那个~~毕竟有很多要更新的~~)
然后就是程序了~~
//------------------------------------------------------------------------------
//4位显示秒表试验
//使用INT0键进行计时的开始和停止,使用S1键作为计时值得清除。定时器T0被用作扫描4
//位数码管(1ms),定时器T1则用来计时(10ms)
#include"ioavr.h"
#include"intrinsics.h"
typedef unsigned char uchar;
typedef unsigned int uint;
__flash uchar seg[16]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,
0x7c,0x39,0x5e,0x79,0x71}; //共阴极数码管
__flash uchar act[4]={0xfe,0xfd,0xfb,0xf7};
uint cnt; // 全局变量,计时值
uchar flag_start; //全局变量,秒表启动标志
uchar i; //每次进入T0,对第i个数码管刷新
#define CPL_BIT(x,y) (x^=(1< //就是将x所对应的第y位取反
#define s1 (PIND&0X10) //对S1进行按键检测的时候用到,如果检测到,对cnt清0,你懂得why~~
#define xtal 8 //由于这里用到的是8MHz的,所以定义为8,写延时函数的时候会用到,回头我会把IAR for AVR的精确延时写到博客上~~
//------------------------------------------------------------------------------
//延时1ms的函数,没有参数传递
void delay_1ms()
{
uint i;
for(i=1;i<(uint)(xtal*143-2);i++)
;
}
//------------------------------------------------------------------------------
//延时nms的函数,有参数传递
void delay_nms(uint n)
{
uint i=0;
while(i {
delay_1ms();
i++;
}
}
//------------------------------------------------------------------------------
//端口定义函数,用来定义ABCD四个端口的输入输出
void port_init()
{
DDRA=0XFF;
PORTA=0X00;
DDRC=0XFF;
PORTC=0XFF;
DDRD=0X00;
PORTD=0XFF;
}
//------------------------------------------------------------------------------
//T0的初始化定义
void timer0_init()
{
TCNT0=0X83; //1ms的定时初值
TCCR0=0X03; //T0的计数预分频取64,这就打开了T0的计数功能
}
//------------------------------------------------------------------------------
//T1的初始化函数
void timer1_init()
{
TCNT1H=0XD8; //10ms的定时初值
TCNT1L=0XF0;
}
//------------------------------------------------------------------------------
//按键检测函数
void scan_s1()
{
if(s1==0)
{
delay_nms(10);
if(s1==0)
cnt=0;
}
}
//------------------------------------------------------------------------------
//芯片初始化函数,包含上面的所有初始化,以及没有涉及的
void device_init()
{
__disable_interrupt(); //disable all interrupts,呵呵,不解释都知道,SREG?这个我不知道在不在里面,从下面的来看应该在,回头会具体分析的~~哈
port_init();
timer0_init();
timer1_init();
MCUCR=0X00; //INT0为低电平时产生中断请求
GICR=0X40; //使能INT0
TIMSK=0X05; //使能T0和T1溢出中断
__enable_interrupt(); //re-enable interrupts
}
//------------------------------------------------------------------------------
//main
void main()
{
device_init();
while(1)
{
if(flag_start==0x01)
TCCR1B=0X02; //如果启动标志位为1,则启动T1,计数预分频取8
if(flag_start==0x00)
{
TCCR1B=0X00; //相反,定时器标志位为0,则关闭T1,停止计数,检测S1,如果按下,对cnt清0
scan_s1();
}
}
}
//------------------------------------------------------------------------------
//INT0
#pragma vector=INT0_vect
__interrupt void int0()
{
CPL_BIT(flag_start,0); //取反启动标志,这里对第0位取反
delay_nms(10);
}
//------------------------------------------------------------------------------
//TIMER0_OVF
#pragma vector=TIMER0_OVF_vect
__interrupt void timer0_ovf()
{
SREG=0X80;
TCNT0=0X83;
if(++i>3)
i=0;
switch(i)
{
case 0: PORTA=seg[cnt];PORTC=act[i];break;
case 1: PORTA=seg[cnt0/10];PORTC=act[i];break;
case 2: PORTA=seg[cnt00/100];PORTC=act[i];break;
case 3: PORTA=seg[cnt/1000];PORTC=act[i];break;
default: break;
}
}
//------------------------------------------------------------------------------
//TIMER1_OVF
#pragma vector=TIMER1_OVF_vect
__interrupt void timer1_ovf()
{
TCNT1H=0XD8;
TCNT1L=0XF0;
if(++cnt>999)
cnt=0;
}


上一页 1 2 3 4 下一页

评论


技术专区

关闭