keil 51 使用c语言的中断向量

上传人:wm****3 文档编号:43228272 上传时间:2018-06-05 格式:DOC 页数:10 大小:18.38KB
返回 下载 相关 举报
keil 51 使用c语言的中断向量_第1页
第1页 / 共10页
keil 51 使用c语言的中断向量_第2页
第2页 / 共10页
keil 51 使用c语言的中断向量_第3页
第3页 / 共10页
keil 51 使用c语言的中断向量_第4页
第4页 / 共10页
keil 51 使用c语言的中断向量_第5页
第5页 / 共10页
点击查看更多>>
资源描述

《keil 51 使用c语言的中断向量》由会员分享,可在线阅读,更多相关《keil 51 使用c语言的中断向量(10页珍藏版)》请在金锄头文库上搜索。

1、keilkeil 5151 使用使用 C C 语言的中断向量语言的中断向量中断源的矢量位置中断源 Keil 中断编号 矢量地址最高优先级 6 0x0033外部中断 0 0 0x0003定时器 0 溢出 1 0x000B外部中断 1 2 0x0013定时器 1 溢出 3 0x001B串口 4 0x0023定时器 2 溢出 5 0x002BDMA 7 0x003B硬件断点 8 0x0043JTAG 9 0x004B软件断点 10 0x0053监视定时器 12 0x0063C 语言在 8051 单片机上的扩展(interrupt、using 关键字的用法)(2008-06-26 14:12:36)转

2、载标签:interruptusingc 语言 it C 语言在 8051 单片机上的扩展(interrupt、using 关键字的用法)直接访问寄存器和端口定义sfr P0 0x80sfr P1 0x81sfr ADCON; 0xDEsbit EA 0x9F操作ADCON = 0x08 ; P1 = 0xFF ; io_status = P0 ;EA = 1 ; 在使用了 interrupt 1 关键字之后,会自动生成中断向量在 ISR 中不能 与其他 “后台循环代码“(the background loop code) 共享局部变量因为 连接器 会复用 在 RAM 中这些变量的 位置 ,所以

3、它们会有不同的意义,这取决于当前使用的不同的函数复用变量对 RAM 有限的 51 来将很重要。所以,这些函数希望按照一定的顺序执行 而不被中断。timer0_int() interrupt 1 using 2unsigned char temp1 ;unsigned char temp2 ;executable C statements ;“interrupt“声明 表示 向量生成在 (8*n3),这里,n 就是interrupt 参数后的那个数字这里,在 08H 的代码区域 生成 LJMP timer0_int 这样一条指令“using“ tells the compiler to swit

4、ch register banks on entry to an interrupt routine. This “context“ switch is the fastest way of providing a fresh registerbank for an interrupt routines local data and is to be preferred to stacking registers for very time-critical routines. Note that interrupts of the same priority can share a regi

5、ster bank, since there is no risk that they will interrupt each other.using 告诉编译器 在进入中断处理器 去切换寄存器的 bank。这个“contet“切换是为中断处理程序的局部变量提供一个新鲜的寄存器 bank 最快的方式。对时序要求严格的程序,是首选的 stack 寄存器(保存寄存器到 stack)方式。注意:同样优先级别的中断 可以共享 寄存器 bank,因为他们每次将中断 没有危险If a USING 1 is added to the timer1 interrupt function prototype,

6、the pushing of registers is replaced by a simple MOV to PSW to switch registerbanks. Unfortunately, while the interrupt entry is speeded up, the direct register addressing used on entry to sys_interp fails. This is because C51 has not yet been told that the registerbank has been changed. If no worki

7、ng registers are used and no other function is called, the optimizer eliminiates teh code to switch register banks.如果在 timer1 的中断函数原型中使用 USING 1, 寄存器的 pushing将被 MOV to PSW 切换寄存器 bank 所替换。不幸的是,当一个中断入口被加速时。用在入口的 直接寄存器寻址将失败。这是因为 C51 没有告诉 寄存器 bank 已经改变。如果不工作的寄存器将被使用,如果没有其他函数被调用,优化器.Logically, with an in

8、terrupt routine, parameters cannot be passed to it or returned. When the interrupt occurs, compiler-inserted code is run which pushes the accumulator, B,DPTR and the PSW (program status word) onto the stack. Finally, on exiting the interrupt routine, the items previously stored on the stack are rest

9、ored and the closing “ causes a RETI to be used rather than a normal RET.逻辑上,一个中断服务程序,不能传递参数进去,也不可返回值。当中断发生时,编译器插入的代码 被运行,它 将 累加器,B,DPTR 和 PSW(程序状态字)入栈。最后,在退出中断程序时,预先存储在栈中 被恢复。最后的“结束符号将 插入 RETI 到 中断程序的最后,为了用 KeilC语言创建一个中断服务程序(ISR) ,利用 interrupt 关键词和正确的中断号声明一个 static void 函数。KeilC编译器自动生成中断向量,以及中断程序的进

10、口、出口代码。Interrupt 函数属性标志着该函数为 ISR。可用 using 属性指定 ISR 使用哪一个寄存器区,这是可选的。有效的寄存器区范围为1 到 3。中断源的矢量位置中断源 Keil 中断编号 矢量地址最高优先级 6 0x0033外部中断 0 0 0x0003定时器 0 溢出 1 0x000B外部中断 1 2 0x0013定时器 1 溢出 3 0x001B串口 4 0x0023定时器 2 溢出 5 0x002BDMA 7 0x003B硬件断点 8 0x0043JTAG 9 0x004B软件断点 10 0x0053监视定时器 12 0x00631.函数在调用前定义与在调用后定义产

11、生的代码是有很大差别的(特别是在优化级别大于 3 级时) 。 (本人也不太清楚为什么,大概因为在调用前定义则调用函数已经知道被调用函数对寄存器的使用情况,则可对函数本身进行优化;而在调用后进行定义则函数不知被调用函数对寄存器的使用情况,它默认被调用函数对寄存器(ACC、 B、 DPH、 DPL、 PSW、 R0、 R1、 R2、 R3、R 4、 R5、, R6、 R7)都已经改变,因此不在这些寄存器中存入有效的数据)2.函数调用函数时除在堆栈中存入返回地址之外,不在堆栈中保存其它任何寄存器(ACC、 B、 DPH、 DPL、 PSW、 R0、 R1、 R2、 R3、R 4、 R5、, R6、

12、R7)的内容。 (除非被调用函数使用了using 特性)3.中断函数是一个例外,它会计算自身及它所调用的函数对寄存器(ACC、 B、 DPH、 DPL、 PSW、 R0、 R1、 R2、 R3、R 4、 R5、, R6、 R7)的改变,并保存相应它认为被改变了的寄存器。4.使用 C 写程序时,尽量少使用 using n (n=0,1,2,3)特性。 (这个特性在本人使用的过程中存在一些问题,不知算不算是一个小 bug)默认 keil c51 中的函数使用的是 0 寄存器组,当中断函数使用using n 时,n = 1,2,3 或许是对的,但 n=0 时,程序就已经存在了bug(只有中断函数及其

13、所调用的函数并没有改变 R0 - R7 的值时,这个 bug 不会表现出来) )一个结论是,在中断函数中如果使用了 using n,则中断不再保存R0-R7 的值。由此可以推论出,一个高优先级的中断函数及一个低优先级的中断函数同时使用了 using n, (n = 0,1,2,3)当 n 相同时,这个存在的 bug 是多么的隐蔽。 (这恰是使人想象不到的)使用不同寄存器组的函数(特殊情况外)不能相互调用using“关键字告诉 编译器 切换 register bank如果中断程序不重要,using 关键字 能忽略。如果一个函数被从中断程序调用,而此中断强制使用 using当编译一个被调用的函数时,编译器必须告诉它1)在函数前 必须用伪 指令#pragma NOAREGS在进入 函数#pragma RESTORE或者#pragmas AREGS这样就不会使用 “绝对地址定位“2)#pragma REGISTERBANK(n)用这个指定告诉当前使用的 bank用 NOAREGS 指令 移除 MOV R7,AR7中断服务例程timer0_int() interrup

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 生活休闲 > 社会民生

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号