受保护的硬件中断处理程序卡住了吗? (DJGPP)

时间:2017-02-08 11:09:25

标签: c dos djgpp

我正在尝试在保护模式下设置硬件中断处理程序,使用进行编译。这是可能的最小代码(定时器中断),我想:

#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)。 我错过了一条线吗?

1 个答案:

答案 0 :(得分:0)

您需要链接旧的中断处理程序,就像文档中链接到的示例Jonathon Reinhart一样,因为旧的处理程序将告诉中断控制器停止断言中断。它还具有保持BIOS时钟滴答的额外好处,因此每次运行程序时都不会丢失几秒钟。否则当你的中断处理程序返回时,CPU会立即再次调用处理程序,你的程序将陷入无限循环。

此外,无法保证GCC会在endHandler之后放置handler。我建议只是简单地锁定页面handler开始和下一页,以防它跨越页面:

_go32_dpmi_lock_code((void *) handler, 4096);

注意这里需要强制转换,因为没有从指针到函数类型的自动转换到指向void的指针。