我正在尝试在保护模式下设置硬件中断处理程序,使用djgpp-2进行dosbox-0.74编译。这是可能的最小代码(定时器中断),我想:
#include <dpmi.h>
#include <go32.h>
#include <stdio.h>
unsigned int counter = 0;
void handler(void) {
++counter;
}
void endHandler(void) {}
int main(void) {
_go32_dpmi_seginfo oldInfo, newInfo;
_go32_dpmi_lock_data(&counter, sizeof(counter));
_go32_dpmi_lock_code(handler, endHandler - handler);
_go32_dpmi_get_protected_mode_interrupt_vector(8, &oldInfo);
newInfo.pm_offset = (int) handler;
newInfo.pm_selector = _go32_my_cs();
_go32_dpmi_allocate_iret_wrapper(&newInfo);
_go32_dpmi_set_protected_mode_interrupt_vector(8, &newInfo);
while (counter < 3) {
printf("%u\n", counter);
}
_go32_dpmi_set_protected_mode_interrupt_vector(8, &oldInfo);
_go32_dpmi_free_iret_wrapper(&newInfo);
return 0;
}
请注意,我没有链接我的处理程序,而是替换它。计数器不会增加超过1
(因此永远不会停止主循环)让我猜测处理程序没有正确返回或只调用一次。另一方面链接工作正常(删除包装行并用set_protected_mode
替换chain_protected_mode
)。
我错过了一条线吗?
答案 0 :(得分:0)
您需要链接旧的中断处理程序,就像文档中链接到的示例Jonathon Reinhart一样,因为旧的处理程序将告诉中断控制器停止断言中断。它还具有保持BIOS时钟滴答的额外好处,因此每次运行程序时都不会丢失几秒钟。否则当你的中断处理程序返回时,CPU会立即再次调用处理程序,你的程序将陷入无限循环。
此外,无法保证GCC会在endHandler
之后放置handler
。我建议只是简单地锁定页面handler
开始和下一页,以防它跨越页面:
_go32_dpmi_lock_code((void *) handler, 4096);
注意这里需要强制转换,因为没有从指针到函数类型的自动转换到指向void的指针。