如何使用独立的 x64/x86 程序集文件构建 x64/x86 项目
1) 启动 Visual Studio (Community) 2017 并选择 FILE - New - Project。
2) 在下一个窗口中选择Empty Project。
3) 确保项目在解决方案资源管理器中突出显示,然后选择 PROJECT - Build Customizations...。
4) 在下一个窗口中勾选masm(.targets,.props) 并点击OK。
5) 从菜单中选择PROJECT - Add New Item。
6) 在下一个窗口中选择 C++File(.cpp) 和 - 重要提示! - 给它一个带有 .asm 扩展名的名称。点击Add。
7) 现在您可以用内容填充文件了。
Source.asm:
EXTERN GetStdHandle : PROC
EXTERN WriteFile : PROC
EXTERN ExitProcess : PROC
.DATA?
hFile QWORD ?
BytesWritten DWORD ?
.DATA
hello BYTE 'Hello world!', 13, 10
.CODE
main PROC
; https://blogs.msdn.microsoft.com/oldnewthing/20160623-00/?p=93735
sub rsp, 40 ; Shadow space (4 * 8) & 1 parameter (8 bytes)
; https://docs.microsoft.com/en-us/cpp/build/stack-allocation
and spl, -16 ; Align to 16
; https://msdn.microsoft.com/library/windows/desktop/ms683231.aspx
mov ecx, -11 ; DWORD nStdHandle = STD_OUTPUT_HANDLE
call GetStdHandle ; Call WinApi
mov hFile, rax ; Save returned handle
; https://msdn.microsoft.com/library/windows/desktop/aa365747.aspx
mov rcx, hFile ; HANDLE hFile (here: Stdout)
lea rdx, hello ; LPCVOID lpBuffer
lea r9, BytesWritten ; LPDWORD lpNumberOfBytesWritten
mov r8d, LENGTHOF hello ; DWORD nNumberOfBytesToWrite
mov qword ptr [rsp+32], 0 ; LPOVERLAPPED lpOverlapped = NULL
call WriteFile ; Call WinAPI
exit:
; https://msdn.microsoft.com/library/windows/desktop/ms682658.aspx
xor ecx, ecx ; Set RCX to null for return value
call ExitProcess ; Call WinAPI to exit
main ENDP
end
这是一个 64 位控制台应用程序,从过程 main 开始。
8) 将 Solution Platforms 更改为 x64
9) 选择PROJECT - Properties。
10) 在“属性”窗口中,您必须完成两个链接器选项:
- 入口点:主要
- 子系统:控制台 (/SUBSYSTEM:CONSOLE)
在左侧选择Configuration Properties - Linker - All Options,同时更改两个选项,然后单击OK。
11) 使用 CTRL-F5 构建并运行 .exe。应用程序将在新窗口中打开。
现在用 32 位控制台应用程序覆盖 Source.asm:
.MODEL flat, stdcall
; https://docs.microsoft.com/en-us/cpp/assembler/masm/proto
GetStdHandle PROTO STDCALL, ; https://docs.microsoft.com/en-us/windows/console/getstdhandle
nStdHandle: SDWORD
WriteFile PROTO STDCALL, ; https://docs.microsoft.com/en-us/windows/desktop/api/fileapi/nf-fileapi-writefile
hFile: DWORD, ; output handle
lpBuffer: PTR BYTE, ; pointer to buffer
nNumberOfBytesToWrite: DWORD, ; size of buffer
lpNumberOfBytesWritten: PTR DWORD, ; num bytes written
lpOverlapped: PTR DWORD ; ptr to asynchronous info
ExitProcess PROTO STDCALL, ; https://docs.microsoft.com/en-us/windows/desktop/api/processthreadsapi/nf-processthreadsapi-exitprocess
dwExitCode: DWORD ; return code
.DATA ; https://docs.microsoft.com/en-us/cpp/assembler/masm/dot-data
Hallo db "Hello world!",13,10
.DATA? ; https://docs.microsoft.com/en-us/cpp/assembler/masm/dot-data-q
lpNrOfChars dd ?
.CODE ; https://docs.microsoft.com/en-us/cpp/assembler/masm/dot-code
main PROC ; docs.microsoft.com/en-us/cpp/assembler/masm/proc
invoke GetStdHandle, -11 ; -> StdOut-Handle into EAX
invoke WriteFile, eax, OFFSET Hallo, LENGTHOF Hallo, OFFSET lpNrOfChars, 0
invoke ExitProcess, 0
main ENDP
END main ; https://docs.microsoft.com/en-us/cpp/assembler/masm/end-masm
将Solution Platforms更改为x86(上面的第8个)并用SubSystem: Console (/SUBSYSTEM:CONSOLE)完成项目属性(上面的第10个)。您不得设置入口点,因为 ml32 需要 END 指令 (END main) 之后的入口点。使用 CTRL-F5 构建并运行 .exe。