调试使用 OpenWatcom 编译的崩溃的 32 位 DOS 可执行文件

Jir*_*oda 5 c windows debugging dos watcom

我有一个使用 OpenWatcom classic(1.9 - 最新稳定版本)为 32 位 DOS/DPMI 目标(带有 DOS32/A 扩展器)编译的应用程序。如果程序因内存访问错误而崩溃,我会得到错误指令的 CS:EIP。如何将其映射到汇编代码/源代码行号?(注:我在 Wine 下使用 Windows 版本的 OpenWatcom(在 Linux 中运行),然后在 DosBox 中运行可执行文件。)

使用 GCC/binutils,我将使用 -ggdb 进行编译,然后在可执行文件上使用 objdump -DS 来获取程序集和源代码视图。有 OpenWatcom 同等产品吗?或者,也许,一个可以做同样事情的交互式调试器?我尝试使用 wdis,但这只适用于目标文件,不适用于可执行文件。由于使用目标文件,我无法知道它将被重新定位到哪里,因此它毫无用处。或者也许至少有一种方法可以为可执行文件生成符号映射?

Mar*_*chi 0

请注意,DOSBox并不完全模拟CPU,尤其是在保护模式调试支持方面。因此,如果您想调试 DOS 保护模式可执行文件,您需要使用 VM 或其他模拟器。

也就是说,您可以执行以下操作。

确保您已设置这些环境变量(假设开发工具路径为C:\WATCOM):

SET PATH=C:\WATCOM\BINW;%PATH%
SET INCLUDE=C:\WATCOM\H
SET WATCOM=C:\WATCOM
SET EDPATH=C:\WATCOM\EDDAT
SET WD=/TR#RSI/SWAP
Run Code Online (Sandbox Code Playgroud)

WD是指定 DOS Watcom 调试器默认选项的选项:

  • /TR#RSI标志指定可执行文件使用 DOS/4G DOS 扩展器
  • /SWAP指定视频内存交换是使用单个页面完成的,如果您正在开发图形应用程序,则这是强制性的。

据我所知WD不支持DOS32/A DOS扩展器,所以你可能决定使用DOS/4G。

请务必指定-d2编译器 ( wcc386) 和debug all链接器 ( wlink) 的标志。

示例生成文件:

LINK_FLAGS_DBG = debug all SYS dos4g op m op maxe=25 op q op symf

CC = wcc386
CC_FLAGS_DBG = -i=C:\WATCOM\H -w4 -e25 -zq -otexan -d2 -5s -bt=dos -mf

OBJS = test.obj

test.exe : $(OBJS) test.lnk
    wlink $(LINK_FLAGS_DBG) @$^*
    
test.lnk : $(OBJS)
    echo NAME $^& >$^@
    echo DEBUG all >>>>$^@
    for %i in ($(OBJS)) do echo FILE %i >>$^@
    
clean :
    del *.obj
    del *.exe
    del *.lnk
    del *.map
    del *.sym

.c.obj : .AUTODEPEND
    $(CC) $[* $(CC_FLAGS_DBG)
Run Code Online (Sandbox Code Playgroud)

测试.c文件:

#include <stdio.h>

void main(int argc, char *argv[]) {
    int test = 1234;

    printf("Hello world!\ntest is %d", test);
}
Run Code Online (Sandbox Code Playgroud)

使用以下命令构建可执行文件(以及符号和映射文件):

wmake

使用以下命令启动 Watcom DOS 调试器:

wd test

您应该在这个屏幕上:

在此输入图像描述

从这里您可以交互式地调试您的程序,就像使用现代调试器一样。

作为旁注:

  • 参考文档,虽然我只安装了DOS工具,但是非常准确和完整
  • 请注意,知道错误指令的地址并不等于知道该指令导致程序崩溃的原因,这就是为什么我邀请您使用(也许是暂时的)DOS/4G 和 Watcom 调试器。