|
最近看程序,发现写不来中断程序,后来发现此篇文章,原来是我学习AVR的书版本太旧,讲解的是老版ACRGCC,这里我把这篇文章转过来,方便大家学习。多个网站上都有这篇文章,具体出处不详。
另外,我已经上传了AVRGCC用户手册到网站,你可以通过 http://www.arduino.cn/avrgcc/ 访问
早期版本的avr-libc 对中断服务程序的书写提供了两个宏SIGNAL 和INTERRUPT,并且需要包含两个头文件:avr/signal.h 和avr/interrupt.h。新版(如2007 版WINAVR)中,INTERRUPT 宏不再可用,而建议用ISR 宏替代SIGNAL宏,ISR 和SIGNAL 是一回事,但以后的版本中SIGNAL 宏将会逐渐被丢弃,所以新的程序建议使用ISR,也就是使用ISR作为中断服务函数名,下面将举例说明一些具体的中断使用。
一.为什么没有了INTERRUPT?
INTERRUPT 宏是各中断嵌套有关的,当中断程序得到执行时AVR 的硬件将全局的中断允许标记清除,其它中断无法再发生,当中断程序退出时自动被允许。而INTERRUPT宏让编译器所做的就是在中断程序的入口处插入一个SEI 指令,退出处插入一个CLI 指令,使得中断可以嵌套,也许这对于普通的程序意义不大,INTERRUPT 才被去掉的。然而仍然可以使用下面的方式实现INTERRUPT 宏的功能:
[mw_shl_code=c,true]void XXX_vect(void) __attribute__((interrupt));
void XXX_vect(void)
{
//程序
}[/mw_shl_code]
即上面方式定义的中断程序可以再次被中断,XXX_vect 是中断例程名字,下面将说明。
二.可使用第二套中断名
早期版本的avr-libc 中中断例程名为SIG_开头,从avr-libc 1.4.0 版开始第二套中断名称,它以_vect 为后缀。因为在器件手册里用比如用TIMER2 COMP表示定时器2比较匹配中断,新的方法更接近手册,称可查libc 手册或器件对应io.h 文件。
三.用户未定义中断的截获
下如书中所说早期版本中用户未定义服务程序的中断发生时系统就会复位,而新版本可以截获这种中断了,使用如下:
[mw_shl_code=c,true]#include <avr/interrupt.h>
ISR(BADISR_vect){
//服务程序
}[/mw_shl_code]
这个服务程序是可选的,并不是强制用户处理未定义中断,如果不定义那结果就是产生未定义中断时复位。因为发生未定义中断往往就是用户程序的错误。
四.中断替换
当两个中断使用同一个服务程序时可例用中断替换宏ISR_ALIAS(vector,target_vector)此时服务程序只写一个。例如:
[mw_shl_code=c,true]#include
ISR(INT0_vect)
{
PORTB = 42;
}
ISR_ALIAS(INT1_vect, INT0_vect);[/mw_shl_code]
当INT1 中断发生时就会执行INT0 的中断程序。Avr-libc 建议不要大量使用这个宏。用ISR实现中断嵌套也很简单
[mw_shl_code=c,true]ISR(INT0_vect,ISR_NOBLOCK)
{
//服务程序
}[/mw_shl_code]
综上所述,新版GCCAVR在一些普通的实际应用中,我们使用中断的基本模式如下:
[mw_shl_code=c,true]#include <avr/interrupt.h>
ISR(xxx_vect){
// 服务程序
}[/mw_shl_code]
其中xxx_vect是对应器件的中断向量,在头文件iomxx.h中有说明,比如器件ATmega16对应的是iom16.h。
关于中断的帮助,用户在编译程序时可以查看对应的help文件avr-libc-user-manual,可以通过AVR GCC的help进入。
|
|