hrefspace

 找回密码
 立即注册
搜索
热搜: PHP PS 程序设计
查看: 690|回复: 0

AVR外部中断

[复制链接]

275

主题

454

帖子

1014

积分

大司空

Rank: 5Rank: 5

积分
1014
发表于 2024-3-22 22:56:01 | 显示全部楼层 |阅读模式
1. 开发语言     本范例使用 WinAVR/GCC 20050214 版本开发2. 范例描述     按下按键0,LED0亮。直到松手,其他按键才能起作用
     按下按键1,LED1亮。其他按键随时都能起作用
     按下按键2,LED0/1都熄灭。直到松手,其他按键才能起作用

3. 电路图设计 :
为简化线路设计,使用了本网站的ATmega16功能小板
.
4. 代码设计与说明 :
/*************************************************
****      AVR 外部中断使用范例                 ***
****                                               ***
**** 策划、整理与测试: 阿莫(ARMok)            ******* 代码设计:  HJJourAVR                     ***
**** 编译器:WINAVR20050214                    ***
****          www.OurAVR.com         2005.8.31           ***
*************************************************/
/*
本程序简单的示范了如何使用ATMEGA16的外部中断
        中断的设置
        按键的简单延时防抖动
        中断的嵌套
        变量在中断中的应用---如果变量会在中断服务程序中被修改,须加volatile限定
        
本范例可直接使出厂状态的新M16芯片,无需对芯片的熔丝位进行配置。
出于简化程序考虑,各种数据没有对外输出,学习时建议使用JTAG ICE硬件仿真器
        
关于外部中断作唤醒源的条件:(将会在后面的电源管理和睡眠模式范例中应用)
    而INT0和INT1的边沿触发中断只能在 空闲模式起作用,即 CLKI/O不停止
    INT0和INT1的低电平中断,INT2在各种睡眠模式下都可以,因为这几种中断工作    于异步模式,不需要时钟驱动
   
官方的M16中文手册对外部中断的描叙存在多处错误,请参考英文原版。*/
  1. #include <avr/io.h>#include <avr/delay.h>#include <avr/signal.h>#include <avr/interrupt.h>/*宏INTERRUPT 的用法与SIGNAL 类似,区别在于SIGNAL 执行时全局中断触发位被清除、其他中断被禁止INTERRUPT 执行时全局中断触发位被置位、其他中断可嵌套执行另外avr-libc 提供两个API 函数用于置位和清零全局中断触发位,它们是经常用到的。分别是:void sei(void) 和void cli(void) 由interrupt.h定义 *///注: 内部函数_delay_ms() 最高延时 262.144mS@1MHz/* 该函数可以实现较精确的定时,但用JTAG仿真时较麻烦---会进入机器码窗口(Disassembeler).注意跳开该语段。一旦JTAG仿真进入该内部函数语句,会变得像"死机"一样(其实在运行中),可以先[break],然后在后面的C语句设[breakpoint],[RUN]跳过*/// ?for()/while()语句计算延时时间较麻烦。// 为了使 _delay_ms()函数的延时正确,须在makefile中设定F_CPU为实际的系统时钟频// 本范例为1MHz内部RC振荡器 即 F_CPU=1000000 /*C:\WinAVR\avr\include\avr\目录包括所有芯片的定义和其他头文件其中iom16.h 定义ATMEGA16芯片的特性(中断向量,寄存器,位定义...)包括下面中断服务程序的常量 SIG_INTERRUPTx ,PORTx,GICR.....*///管脚定义#define EXT_INT0 2 //PD2 按键0#define EXT_INT1 3 //PD3 按键1#define EXT_INT2 2 //PB2 按键2#define LED0 0 //PB0#define LED1 1 //PB1#define LED2 3 //PB3//宏定义#define LED0_ON() PORTB|= (1<<LED0) //输出高电平,灯亮#define LED0_OFF() PORTB&=~(1<<LED0) //输出低电平,灯灭#define LED1_ON() PORTB|= (1<<LED1)#define LED1_OFF() PORTB&=~(1<<LED1)#define LED2_ON() PORTB|= (1<<LED2)#define LED2_OFF() PORTB&=~(1<<LED2)//51系列的高电平输出能力很弱,低电平也仅能点亮LED.所以常见输出低电平才灯亮的接法。//AVR芯片的高低驱动能力都很强,甚至能推动8字数码管的公共极,怎么接都没问题。//全局变量#define has_volatile 1 //这里是条件编译//可以修改has_volatile=1或0来看程序运行的效果#if has_volatile volatile unsigned char FLAG; //全局变量,会在中断服务程序中被修改,须加volatile限定#elseunsigned char FLAG; //全局变量.#endif//仿真时在watch窗口,监控这些变量。SIGNAL(SIG_INTERRUPT0) //INT0中断服务程序   {      //硬件自动清除INTF0标志位       _delay_ms(10); //延时       if ((PIND&(1<<EXT_INT0))==0) //重复检测,防抖动       LED0_ON(); //点亮LED0       loop_until_bit_is_set(PIND,EXT_INT0); //等待按键释放(变为高电平)       _delay_ms(10); //延时 按键释放时也会抖动。       // 即使同时发生其它的中断事件,如果在这里把相应的中断标志位清除,那么该中断将          不能触发进入中断服务       /*        注意       读端口用 PINx       写端口用 PORTx       */    }INTERRUPT(SIG_INTERRUPT1) //INT1中断服务程序    {       //硬件自动清除INTF1标志位       //这里全局中断被打开,将允许其他中断嵌套执行       _delay_ms(10);       if ((PIND&(1<<EXT_INT1))==0)       LED1_ON(); //点亮LED1       loop_until_bit_is_set(PIND,EXT_INT1);       _delay_ms(10);    }SIGNAL(SIG_INTERRUPT2) //INT2中断服务程序    {       //硬件自动清除INTF2标志位       _delay_ms(10);       if ((PINB&(1<<EXT_INT2))==0)         {            LED0_OFF(); //熄灭LED0            LED1_OFF(); //熄灭LED1         }            loop_until_bit_is_set(PINB,EXT_INT2);            FLAG=!FLAG; //修改全局变量            _delay_ms(100);     }int main(void)    {       //上电默认DDRx=0x00,PORTx=0x00 输入,无上拉电阻       PORTA =0xFF; //不用的管脚使能内部上拉电阻。       PORTC =0xFF;       PORTD =0xFF;       DDRB = (1<<LED2)|(1<<LED1)|(1<<LED0); //输出       PORTB =~((1<<LED2)|(1<<LED1)|(1<<LED0)); //低电平,灯灭       //外部中断INT0,1,2 做按键输入,使能内部上拉,就可以不用外接电阻了       MCUCR=(1<<ISC11)|(0<<ISC10)|(1<<ISC01)|(0<<ISC00); //注意该寄存器有多个功能       /*       ISCx1:0=00 INTx引脚为低电平时产生中断请求       ISCx1:0=01 INTx引脚上任意的逻辑电平变化都将引发中断       ISCx1:0=10 INTx引脚的下降沿产生中断请求       ISCx1:0=11 INTx引脚的上升沿产生中断请求       */       MCUCSR&=~(1<<ISC2); //注意该寄存器有多个功能       /*       ISC2=0 INT2引脚的下降沿产生异步中断请求       ISC2=1 INT2引脚的上升沿产生异步中断请求       */       GIFR=(1<<INTF1)|(1<<INTF0)|(1<<INTF2);//写1清除标志位,在使能中断前最好先把对应                                             // 的标志位清除,以免误触发       GICR=(1<<INT1)|(1<<INT0)|(1<<INT2); //使能三个外部中断       FLAG=0;       sei(); //使能全局中断       while (1)          {             while (FLAG==0);                LED2_ON(); //如果FLAG不加volatile限定(即has_volatile=0),                           //程序将永远都运行不到这里。             while (FLAG!=0);                LED2_OFF();           }     }
复制代码
程序运行效果       按下按键0,LED0亮。直到松手,其他按键才能起作用       按下按键1,LED1亮。其他按键随时都能起作用       按下按键2,LED0/1都熄灭。直到松手,其他按键才能起作用              LED2是根据按键2的顺序来亮灭,松手后变换,前提是FLAG加了volatile限定

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?立即注册

x
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|小黑屋|hrefspace

GMT+8, 2025-1-8 13:11 , Processed in 0.060020 second(s), 22 queries .

Powered by hrefspace X3.4 Licensed

Copyright © 2022, hrefspace.

快速回复 返回顶部 返回列表