我需要生成一个跳远指令来跳转到另一个ISR(中断服务例程)。我正在开发一个32位的FreeDOS应用程序。
在阅读了OW手册(cguide.pdf和clr.pdf)之后,我想出了两种成功编译任何警告或错误的方法。
/* Code Snippet #1 */
#pragma aux old08 aborts ;
void (__interrupt __far *old08)(void); // function pointer declaration
void __interrupt __far new08(void) {
/* Do some processing here ... */
(*old08)(); /* OW will now generate a jump inst. instead of call*/
}我发现的另一个方法是:
/* Code Snippet #2 */
static void jumpToOld08(void);
# pragma aux jumpToOld08 = \
".686p" \
" DB 0xEA" \
"off_old08 DD 0" \
"sel_old08 DW 0" ;
void __interrupt __far new08(void){
/* Do some processing here ... */
jumpToOld08();
}
extern unsigned short sel_old08;
extern unsigned int off_old08;
sel_old08 = ( __segment )FP_SEG(old08);
off_old08 = FP_OFF(old08); 现在我的问题是,以上两种方法中哪一种更正确或更好?有什么想法或意见吗?
还有其他方法可以做到这一点吗?
发布于 2012-07-28 03:51:05
#include <dos.h>
void _chain_intr( void (__interrupt __far *func)(void) );此函数可用于跳转到链中的另一个中断处理程序。这个函数永远不会返回。它弹出由中断关键字保存的所有寄存器并跳转到处理程序。当func指定的中断处理程序接收到控制时,堆栈和寄存器就会显示为中断刚刚发生。
此函数只能在使用中断关键字声明的函数中使用。
跳转而不是调用的好处在irq处理程序中并不明显,但是对于软件中断处理程序来说绝对是如此。链中的下一个软件中断处理程序期望cpu寄存器包含一些信息,例如传递给它的参数,因此在跳转到下一个处理程序之前,chainintr恢复所有cpu寄存器,就好像下一个处理程序直接接收控件一样。
发布于 2012-08-19 09:00:41
我可能会写这样的东西:
void far_jump (uint32_t offset, uint16_t selector)
{
/* remove the (callee's) stack frame including the return address by manipulating esp & ebp */
/* the top of the stack now points to offset:selector */
_asm
{
retf ; or whatever the asm syntax dictates
}
/* esp & ebp will now point to the caller's stack frame */
}据我回忆,32位模式将推送和弹出选择器作为32位单元,即使只使用低16位。
https://stackoverflow.com/questions/11538081
复制相似问题