【问题标题】:How to write and combine multiple source files for a project in MASM?如何在 MASM 中为一个项目编写和组合多个源文件?
【发布时间】:2020-06-28 03:54:12
【问题描述】:

对组装很陌生,玩得很开心。我想将我的程序的功能拆分到多个文件中,特别是通过将类似的功能组合在一起进行组织。这些其他文件将由主文件调用(希望甚至是其他非主文件)。我还没有做到这一点,希望得到帮助。

我没有使用 IDE,我更喜欢使用 notepad++、ml.exe 和 link.exe(来自 MASM 文件夹)来自己编写、组装和链接程序。我看过的大多数在线资源都假设 Visual Studio,并提供对我不起作用的代码,或者可能是不完整的 b/c IDE 做其他事情。我不打算开始使用 IDE。

我想学习“最佳”方式,即对未来项目最有用的方式。我可以这样设置它,以便我可以复制文件并编写几行代码以便将来在不同的项目中使用它吗?或者也许这是不好的做法,我应该学习一种更标准的方法?我知道这个平台不适用于有观点的问题,我希望这个问题比观点更基于事实。

我能想到的所有有用信息:
语言:Masm 汇编 x86
计算机:64 位 Windows

代码:

运行.bat

@echo off

ml /c /coff /Zi /Fl Driver.asm
ml /c /coff /Zi /Fl Utils.asm

link /debug /subsystem:console /entry:start /out:Driver.exe Utils.obj Driver.obj \masm32\lib\kernel32.lib

Driver.exe

pause

驱动程序.asm

.386
.model flat
.stack 100h

ExitProcess PROTO Near32 STDCALL, dwExitCode:DWORD
ClearRegs PROTO

.DATA
.CODE

PUBLIC _start
_start:

    Main PROC
        MOV EAX, 0
        INVOKE ClearRegs
        INVOKE ExitProcess, 0
    Main ENDP

END

Utils.asm

.386
.model flat
.stack 100h

OPTION PROC:PRIVATE ; Set procedures to private by default

PUBLIC ClearRegs

.DATA
.CODE
    
    ClearRegs PROC C
        XOR EAX, EAX
        XOR EBX, EBX
        XOR ECX, ECX
        XOR EDX, EDX
        XOR ESI, ESI
        XOR EDI, EDI
        RET
    ClearRegs ENDP

END

终端输出

Microsoft (R) Macro Assembler Version 6.14.8444
Copyright (C) Microsoft Corp 1981-1997.  All rights reserved.

 Assembling: Driver.asm
Microsoft (R) Macro Assembler Version 6.14.8444
Copyright (C) Microsoft Corp 1981-1997.  All rights reserved.

 Assembling: Utils.asm
Microsoft (R) Incremental Linker Version 5.12.8078
Copyright (C) Microsoft Corp 1992-1998. All rights reserved.

Driver.obj : error LNK2001: unresolved external symbol ClearRegs
Driver.exe : fatal error LNK1120: 1 unresolved externals
'Driver.exe' is not recognized as an internal or external command,
operable program or batch file.
Press any key to continue . . .

【问题讨论】:

  • 另一种启用代码重用的方法是创建一个库(静态或动态)。这样可以避免在根本不需要更改函数实现的情况下复制源代码。但是在单个项目中,是的,您可以制作多个.asm 文件,分别组装它们,然后将它们链接在一起形成.exe
  • 您是在问如何 这样做,还是在问这是否是个好主意?我假设您可以将多个.obj 文件传递​​给link。您似乎有一个 .bat 文件,其简单模式假定一个 .obj,因此请使用更好的构建系统(如 make),或者如果您使用 *.asm 或其他东西,请改进您的 .bat 以使其正常工作。
  • 我在问如何以及是否要这样做。听从您的建议,我分别组装了每个,效果很好,然后尝试将两个 .obj 都传递给链接,我得到未解析的外部符号(方法的名称)。 Make 似乎是一个叫做 GNU 的东西的一部分,我认为我不想要它(我应该得到它吗?)。我开始研究库,发现MASM文件夹中没有lib.exe,导致发现老师给了我一个更简单的版本。使用 .lib 文件是我应该采用的方式吗?
  • 那些未定义的符号...您是否在声明它们的文件中将它们标记为public,以便在链接完成时它们对其他模块(目标文件)可见?
  • 我相信是的。我已将代码添加到原始问题以及您和其他人的终端输出。

标签: winapi assembly x86 linker masm


【解决方案1】:

现在您的问题已经更新为一个最小的、完整的、可验证的示例,可以确定一些具体问题。当你用PROC 声明一个函数时,每个函数都有一个语言命名和调用约定。不指定一个关联者不做特殊处理。

您可以使用模型指令作为第二个参数指定默认语言。在您使用的两个文件中:

.model flat

所以您没有关联默认语言。您已将ClearRegs 定义为:

ClearRegs PROC C
[snip]
ClearRegs ENDP

这里的问题是PROC C 指定了C 语言的调用约定和命名约定。对于 COFF 格式(32 位),C 命名约定要求在函数名称的开头添加下划线 (_)。如果您要生成一个 MAP 文件,您会发现从 utils.asm 导出的函数名称实际上是 _ClearRegs 而不是 ClearRegs

有多种方法可以解决此问题。您可以选择不向.model 指令添加默认语言,并通过更改告诉Driver.asm ClearRegs 被定义为C PROTOtype:

ClearRegs PROTO

ClearRegs PROTO C

所以现在 utils.asm 正在导出 _ClearRegsDriver.asm 正在导入 _ClearRegs,因为双方都匹配,MASM 将处理添加额外的下划线。 INVOKE ClearRegs 将使用与 PROTO 语句相关的命名约定,该语句表示语言为 C,因此它将为您添加额外的 _

这带来了您可以进行的其他更改。 END 指令可用于指定程序的入口点,而不是在链接器命令行上使用 /entry:<name>。入口点的名称必须以 _ 开头,以满足链接器的要求。

您目前在 Driver.asm 中使用它:

PUBLIC _start
_start:

Main PROC
    [snip]
Main ENDP

END

链接时使用/entry:start。您可以将其更改为:

_Main PROC
    [snip]
_Main ENDP

END _Main   ; END with a function name tells linker to use _Main as program entry point

在链接时,您现在可以完全删除 /entry 选项,并且您不再需要 _start 标签。不过我们可以做得更好。 MS C 运行时启动调用的入口点假定函数遵循 C 语言命名和调用约定。最好这样做:

Main PROC C
    [snip]
Main ENDP

END Main   ; END with a function name tells linker to use _Main as program entry point

如果您打算将所有函数设为PROC C,那么您可以通过更改Utils.asmDriver.asmC /em> 通过改变:

.model flat

到:

.model flat, C

这将更改PROTO 语句的默认值,PUBLIC 语句指定使用PROCPROC 语句本身定义的函数。 Driver.asm 中的代码可能如下所示:

.386
.model flat, C
.stack 100h

ExitProcess PROTO Near32 STDCALL, dwExitCode:DWORD
ClearRegs PROTO

.DATA
.CODE

Main PROC
    MOV EAX, 0
    INVOKE ClearRegs
    INVOKE ExitProcess, 0
Main ENDP

END Main

Utils.asm 可能看起来像:

.386
.model flat, C
.stack 100h

OPTION PROC:PRIVATE ; Set procedures to private by default

PUBLIC ClearRegs

.DATA
.CODE
    
ClearRegs PROC
    XOR EAX, EAX
    XOR EBX, EBX
    XOR ECX, ECX
    XOR EDX, EDX
    XOR ESI, ESI
    XOR EDI, EDI
    RET
ClearRegs ENDP

END

你会链接到:

link /debug /subsystem:console /out:Driver.exe Utils.obj Driver.obj \masm32\lib\kernel32.lib

【讨论】:

  • 非常感谢,这是一个很好的答案。您使我免于使用 lib.exe 并帮助我删除了 _start 标签,这是我真的讨厌使用的东西。几个循环来组装所有 .asm 文件并将它们链接在一起,我的 .bat 文件也很好!先生,真是个书生君子。
猜你喜欢
  • 2013-05-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-05-06
  • 2011-04-20
  • 2021-02-16
相关资源
最近更新 更多