专家
公告
财富商城
电子网
旗下网站
首页
问题库
专栏
标签库
话题
专家
NEW
门户
发布
提问题
发文章
STM32
求大神告诉一下atmega88pa单片机的定时器0/1的中断程序怎么写啊
2019-10-14 20:38
发布
×
打开微信“扫一扫”,打开网页后点击屏幕右上角分享按钮
站内问答
/
STM32/STM8
4935
4
1522
求大神告诉一下atmega88pa单片机的定时器0/1的中断程序怎么写啊!编译器用的是avrstudio 6.0,我搜了一些网上的方法可是都不行
友情提示:
此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
4条回答
乱世疯子狂
2019-10-14 23:26
来俩不甜的 发表于 2017-3-6 19:29
哪里不行说清楚啊。没用过,猜测就是从头文件里找下中断向量,用过mega128就是这样
中断是CPU的一种核心功能,当CPU外部或内部发生指定事件时,就会触发中断,中断当前CPU正在执行的程序,保存当前状态,程序指针跳转中断服务程序的起始地址开始执行.执行完成后,跳回原程序位置恢复中断前的状态.
每个CPU都有一张中断向量表,可以通过CPU的datasheet手册查找。标准C中并没有规定中断函数的写法,所以中断的语法,是各个编译器自己定义的。
VCC编译器的语法规则如下:
#pragma interrupt_handler adc_isr:15
void adc_isr(void)
{
AdcVal = ADC&0x3FF;
ADMUX = (1<<REFS0)|(AdcMux&0x0F); //使用AVcc作为ADC参考电源
ADCSRA |= (1<<ADSC); //ADSC: AD start conversion
}
首先使用“#pragma interrupt_handler adc_isr:15”声明中断
其中“#pragma interrupt_handler”是固定的,表示声明后面紧跟的“adc_isr”函数是中断向量号为15的中断服务函数
然后余下的部分就是编写中断服务函数了。
GCC编译器的语法规则如下:
这里照抄一份网上流传的GCC编译器语法
1.头文件
如果使用GCC-AVR的中断操作,必须包括头文件interrupt.h,即 #include
进一步说明:WinAVR20100110版本已经不支持使用头文件signal.h了,为了向后兼容,如果执意引入signal.h,其实相当于也是引入的interrupt.h。
#ifndef _AVR_SIGNAL_H_
#define _AVR_SIGNAL_H_
//这两句加上最后的#endif是条件定义,防止重复引用的
#warning "This header file is obsolete. Use ."
//这一句【警告】指出,让我们使用interrupt.h
#include
//这一句告诉我们,即使你忘了或者不知道使用interrupt.h,那么也不会出错,引用signal.h相当于引如了interrupt.h。不过何必脱了裤子放屁——多此一举呢,直接引用interrupt.h就好了!
#endif
2.中断函数格式
现在官方推荐的中断书写格式就有一种:
ISR(INTERRUPT_vect)//INTERRUPT_vect是中断向量名称
{
//中断处理
}
3.中断向量的名称
中断向量的名称在WinAVR的最近几个版本中有所变化,最好的办法就是打开查看X:WinAVR-20100110avrincludeavr路径下的你使用的处理器所对应的IO定义头文件。比如我使用的是ATmega16A,我就在此路径下找到了iom16a.h,打开它,找到
这两句话后面的,就是本型号单片机定义的所有中断。以下就是ATmega16A的中断,全部照抄如下:
============================开始====================================
#define INT0_vect_num 1
#define INT0_vect _VECTOR(1)
#define INT1_vect_num 2
#define INT1_vect _VECTOR(2)
#define TIMER2_COMP_vect_num 3
#define TIMER2_COMP_vect _VECTOR(3)
#define TIMER2_OVF_vect_num 4
#define TIMER2_OVF_vect _VECTOR(4)
#define TIMER1_CAPT_vect_num 5
#define TIMER1_CAPT_vect _VECTOR(5)
#define TIMER1_COMPA_vect_num 6
#define TIMER1_COMPA_vect _VECTOR(6)
#define TIMER1_COMPB_vect_num 7
#define TIMER1_COMPB_vect _VECTOR(7)
#define TIMER1_OVF_vect_num 8
#define TIMER1_OVF_vect _VECTOR(8)
#define TIMER0_OVF_vect_num 9
#define TIMER0_OVF_vect _VECTOR(9)
#define SPISTC_vect_num 10
#define SPISTC_vect _VECTOR(10)
#define USARTRXC_vect_num 11
#define USARTRXC_vect _VECTOR(11)
#define USARTUDRE_vect_num 12
#define USARTUDRE_vect _VECTOR(12)
#define USARTTXC_vect_num 13
#define USARTTXC_vect _VECTOR(13)
#define ADC_vect_num 14
#define ADC_vect _VECTOR(14)
#define EE_RDY_vect_num 15
#define EE_RDY_vect _VECTOR(15)
#define ANA_COMP_vect_num 16
#define ANA_COMP_vect _VECTOR(16)
#define TWI_vect_num 17
#define TWI_vect _VECTOR(17)
#define INT2_vect_num 18
#define INT2_vect _VECTOR(18)
#define TIMER0_COMP_vect_num 19
#define TIMER0_COMP_vect _VECTOR(19)
#define SPM_RDY_vect_num 20
#define SPM_RDY_vect _VECTOR(20)
#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.避免在中断内进行数学运算。数学运算将很有可能用到库函数和许多中间变量,就算不出现递归调用的问题,光在中断入口和出口处为了保护和恢复这些中间临时变量就需要大量的开销,严重影响中断服务的效率。
加载中...
查看其它4个回答
一周热门
更多
>
相关问题
STM32F4上I2C(在PROTEUS中模拟)调试不通的问题
6 个回答
芯片供应紧张,准备换个MCU,MM32L系列替换STM32L系列的怎么样?
7 个回答
STM32同时使用两个串口进行数据收发时数据丢包的问题
5 个回答
STM32F103串口通信死机问题
4 个回答
STM32WLE5CC连接SX1268在LoRa模式下能与 SX1278互通吗?
2 个回答
相关文章
ST公司第一款无线低功耗单片机模块有效提高物联网设计生产效率
0个评论
如何实现对单片机寄存器的访问
0个评论
通过USB用STM32片内自带Bootloader下载程序及注意事项
0个评论
欲练此功必先自宫之STM32汇编启动,放慢是为了更好的前行
0个评论
×
关闭
采纳回答
向帮助了您的知道网友说句感谢的话吧!
非常感谢!
确 认
×
关闭
编辑标签
最多设置5个标签!
STM32
保存
关闭
×
关闭
举报内容
检举类型
检举内容
检举用户
检举原因
广告推广
恶意灌水
回答内容与提问无关
抄袭答案
其他
检举说明(必填)
提交
关闭
×
打开微信“扫一扫”,打开网页后点击屏幕右上角分享按钮
×
付费偷看金额在0.1-10元之间
确定
×
关闭
您已邀请
0
人回答
查看邀请
擅长该话题的人
回答过该话题的人
我关注的人
中断是CPU的一种核心功能,当CPU外部或内部发生指定事件时,就会触发中断,中断当前CPU正在执行的程序,保存当前状态,程序指针跳转中断服务程序的起始地址开始执行.执行完成后,跳回原程序位置恢复中断前的状态.
每个CPU都有一张中断向量表,可以通过CPU的datasheet手册查找。标准C中并没有规定中断函数的写法,所以中断的语法,是各个编译器自己定义的。
VCC编译器的语法规则如下:
#pragma interrupt_handler adc_isr:15
void adc_isr(void)
{
AdcVal = ADC&0x3FF;
ADMUX = (1<<REFS0)|(AdcMux&0x0F); //使用AVcc作为ADC参考电源
ADCSRA |= (1<<ADSC); //ADSC: AD start conversion
}
首先使用“#pragma interrupt_handler adc_isr:15”声明中断
其中“#pragma interrupt_handler”是固定的,表示声明后面紧跟的“adc_isr”函数是中断向量号为15的中断服务函数
然后余下的部分就是编写中断服务函数了。
GCC编译器的语法规则如下:
这里照抄一份网上流传的GCC编译器语法
1.头文件
如果使用GCC-AVR的中断操作,必须包括头文件interrupt.h,即 #include
进一步说明:WinAVR20100110版本已经不支持使用头文件signal.h了,为了向后兼容,如果执意引入signal.h,其实相当于也是引入的interrupt.h。
#ifndef _AVR_SIGNAL_H_
#define _AVR_SIGNAL_H_
//这两句加上最后的#endif是条件定义,防止重复引用的
#warning "This header file is obsolete. Use ."
//这一句【警告】指出,让我们使用interrupt.h
#include
//这一句告诉我们,即使你忘了或者不知道使用interrupt.h,那么也不会出错,引用signal.h相当于引如了interrupt.h。不过何必脱了裤子放屁——多此一举呢,直接引用interrupt.h就好了!
#endif
2.中断函数格式
现在官方推荐的中断书写格式就有一种:
ISR(INTERRUPT_vect)//INTERRUPT_vect是中断向量名称
{
//中断处理
}
3.中断向量的名称
中断向量的名称在WinAVR的最近几个版本中有所变化,最好的办法就是打开查看X:WinAVR-20100110avrincludeavr路径下的你使用的处理器所对应的IO定义头文件。比如我使用的是ATmega16A,我就在此路径下找到了iom16a.h,打开它,找到
这两句话后面的,就是本型号单片机定义的所有中断。以下就是ATmega16A的中断,全部照抄如下:
============================开始====================================
#define INT0_vect_num 1
#define INT0_vect _VECTOR(1)
#define INT1_vect_num 2
#define INT1_vect _VECTOR(2)
#define TIMER2_COMP_vect_num 3
#define TIMER2_COMP_vect _VECTOR(3)
#define TIMER2_OVF_vect_num 4
#define TIMER2_OVF_vect _VECTOR(4)
#define TIMER1_CAPT_vect_num 5
#define TIMER1_CAPT_vect _VECTOR(5)
#define TIMER1_COMPA_vect_num 6
#define TIMER1_COMPA_vect _VECTOR(6)
#define TIMER1_COMPB_vect_num 7
#define TIMER1_COMPB_vect _VECTOR(7)
#define TIMER1_OVF_vect_num 8
#define TIMER1_OVF_vect _VECTOR(8)
#define TIMER0_OVF_vect_num 9
#define TIMER0_OVF_vect _VECTOR(9)
#define SPISTC_vect_num 10
#define SPISTC_vect _VECTOR(10)
#define USARTRXC_vect_num 11
#define USARTRXC_vect _VECTOR(11)
#define USARTUDRE_vect_num 12
#define USARTUDRE_vect _VECTOR(12)
#define USARTTXC_vect_num 13
#define USARTTXC_vect _VECTOR(13)
#define ADC_vect_num 14
#define ADC_vect _VECTOR(14)
#define EE_RDY_vect_num 15
#define EE_RDY_vect _VECTOR(15)
#define ANA_COMP_vect_num 16
#define ANA_COMP_vect _VECTOR(16)
#define TWI_vect_num 17
#define TWI_vect _VECTOR(17)
#define INT2_vect_num 18
#define INT2_vect _VECTOR(18)
#define TIMER0_COMP_vect_num 19
#define TIMER0_COMP_vect _VECTOR(19)
#define SPM_RDY_vect_num 20
#define SPM_RDY_vect _VECTOR(20)
#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.避免在中断内进行数学运算。数学运算将很有可能用到库函数和许多中间变量,就算不出现递归调用的问题,光在中断入口和出口处为了保护和恢复这些中间临时变量就需要大量的开销,严重影响中断服务的效率。
一周热门 更多>