中断是CPU的一种核心功能,当CPU外部或内部发生指定事件时,就会触发中断,中断当前CPU正在执行的程序,保存当前状态,程序指针跳转中断服务程序的起始地址开始执行.执行完成后,跳回原程序位置恢复中断前的状态. 每个CPU都有一张中断向量表,可以通过CPU的datasheet手册查找。标准C中并没有规定中断函数的写法,所以中断的语法,是各个编译器自己定义的。 VCC编译器的语法规则如下: #pragma interrupt_handler adc_isr:15 首先使用“#pragma interrupt_handler adc_isr:15”声明中断 其中“#pragma interrupt_handler”是固定的,表示声明后面紧跟的“adc_isr”函数是中断向量号为15的中断服务函数 然后余下的部分就是编写中断服务函数了。 GCC编译器的语法规则如下: 这里照抄一份网上流传的GCC编译器语法 1.头文件 如果使用GCC-AVR的中断操作,必须包括头文件interrupt.h,即 进一步说明:WinAVR20100110版本已经不支持使用头文件signal.h了,为了向后兼容,如果执意引入signal.h,其实相当于也是引入的interrupt.h。 #ifndef _AVR_SIGNAL_H_ #define _AVR_SIGNAL_H_ //这两句加上最后的#endif是条件定义,防止重复引用的 #warning 'This header file is obsolete. //这一句【警告】指出,让我们使用interrupt.h #include //这一句告诉我们,即使你忘了或者不知道使用interrupt.h,那么也不会出错,引用signal.h相当于引如了interrupt.h。不过何必脱了裤子放屁——多此一举呢,直接引用interrupt.h就好了! #endif 2.中断函数格式 现在官方推荐的中断书写格式就有一种: ISR(INTERRUPT_vect)//INTERRUPT_vect是中断向量名称 { //中断处理 } 3.中断向量的名称 中断向量的名称在WinAVR的最近几个版本中有所变化,最好的办法就是打开查看X:\WinAVR-20100110\avr\include\avr路径下的你使用的处理器所对应的IO定义头文件。比如我使用的是ATmega16A,我就在此路径下找到了iom16a.h,打开它,找到 这两句话后面的,就是本型号单片机定义的所有中断。以下就是ATmega16A的中断,全部照抄如下: ============================开始==================================== #define INT0_vect_num #define INT0_vect #define INT1_vect_num #define INT1_vect #define TIMER2_COMP_vect_num #define TIMER2_COMP_vect #define TIMER2_OVF_vect_num #define TIMER2_OVF_vect #define TIMER1_CAPT_vect_num #define TIMER1_CAPT_vect #define TIMER1_COMPA_vect_num #define TIMER1_COMPA_vect #define TIMER1_COMPB_vect_num #define TIMER1_COMPB_vect #define TIMER1_OVF_vect_num #define TIMER1_OVF_vect #define TIMER0_OVF_vect_num #define TIMER0_OVF_vect #define SPISTC_vect_num #define SPISTC_vect #define USARTRXC_vect_num #define USARTRXC_vect #define USARTUDRE_vect_num #define USARTUDRE_vect #define USARTTXC_vect_num #define USARTTXC_vect #define ADC_vect_num #define ADC_vect #define EE_RDY_vect_num #define EE_RDY_vect #define ANA_COMP_vect_num #define ANA_COMP_vect #define TWI_vect_num #define TWI_vect #define INT2_vect_num #define INT2_vect #define TIMER0_COMP_vect_num #define TIMER0_COMP_vect #define SPM_RDY_vect_num #define SPM_RDY_vect #define _VECTOR_SIZE 4 #define _VECTORS_SIZE (21 * _VECTOR_SIZE) =========================结束================================== 就以外部中断向量0为例吧。INT0_vect就是中断向量的名称或者写法了。加粗是我处理的,加粗的就是中断向量名称。 比如,写外部中断0的中断处理函数,就必须这么写: ISR(INT0_vect) { // } 再比如串口接收完成中断函数,就必须这么写: ISR(USARTRXC_vect) { // } 要注意,是USARTRXC_vect,而不是USART_RXC_vect!!! 4.中断函数的设计着力点 使用C语言编写的处理代码,主要考虑中断功能上的处理,而不需要考虑现场保护和恢复等问题。编译器会自动加入代码实现中断现场的保护,并在中断结束时自动恢复现场。但如果在中断服务程序中需要修改某些全局变量时,是否需要保护这些变量的初值将由编程员自己决定和实施。 5.C 语言编写ISR的原则 两个字:高效。 更具体的,体现为: 1.代码尽量简短,中断服务强调的是一个“快”字。(中断处理很“快”,是使用中断而不是查询的重要原因) 2.避免在中断内使用函数调用。虽然 GCC-AVR允许在中断里调用其它函数,但为了避免递归调用的问题,此函数必须为中断服务独家专用。如果非要调用,不妨把原本要写在其它函数内的代码直接写在中断服务程序中。 3.避免在中断内进行数学运算。数学运算将很有可能用到库函数和许多中间变量,就算不出现递归调用的问题,光在中断入口和出口处为了保护和恢复这些中间临时变量就需要大量的开销,严重影响中断服务的效率。 |
|