c++调用汇编中函数的过程
这篇文章探讨,c++调用汇编中函数的过程。核心在于,参数传递过程,会用到固定的寄存器,此处,汇编函数的参数和MessageBoxA 的参数一致所以没有显示传递。
实现的功能:
在汇编的 wechat_asm_call01 函数中调用弹窗MessageBoxA,在c++中引入extern "C" void __stdcall wechat_asm_call01(HWND hWnd, LPCSTR lpText, LPCSTR lpCaption, UINT Type);该函数,然后调用汇编函数并传递参数wechat_asm_call01(0, "内容" , "123123标题", MB_YESNOCANCEL),最终实现c++调用汇编中的弹窗。
函数传参
; https://blog.youkuaiyun.com/a2831942318/article/details/127774181
; 在 Win64 里使用下面寄存器来传递参数:
; rcx - 第 1 个参数
; rdx - 第 2 个参数
; r8 - 第 3 个参数
; r9 - 第 4 个参数
MessageBoxA 参数的结构
调用MessageBoxA过程1,显示一个消息框。
; https://learn.microsoft.com/zh-cn/windows/win32/api/winuser/nf-winuser-messageboxa
; int MessageBoxA(
; [in, optional] HWND hWnd,
; [in, optional] LPCSTR lpText,
; [in, optional] LPCSTR lpCaption,
; [in] UINT uType
;);
c++调用汇编的方法
参照 MSDN 中 MessageBoxA 1 的 api 所需的参数,我们在c++中引用汇编代码中的函数。
// 引入汇编函数
extern "C" void __stdcall wechat_asm_call01(HWND hWnd, LPCSTR lpText, LPCSTR lpCaption, UINT Type);
// 然后在方法中调用
wechat_asm_call01(0, "内容" , "123123标题", MB_YESNOCANCEL);
定义函数,并使用参数
这里,为什么没有显示的接受参数,因为c++传递参数后,正好在汇编中也是用这几个寄存器读取参数,汇编函数的参数和MessageBoxA 的参数一致。所以不需要进行处理。
.code
align 16
extern MessageBoxA : proc
wechat_asm_call01 proc
sub rsp, 28h
call MessageBoxA
add rsp, 28h
ret
wechat_asm_call01 endp
end
参数交换位置
上一篇文章23中,在汇编的结构体(函数)中这样写,就是把第一个参数rcx的值,存如第三个参数r8中。
mov r8, rcx
; 在编写汇编代码时,通常需要为栈分配空间。在调用函数之前,需要将栈指针rsp向下移动,以便为函数的局部变量和参数分配空间。在函数返回之前,需要将栈指针恢复到原来的位置。
; 栈的大小取决于函数的局部变量和参数的数量和大小。在你的例子中,由于没有定义任何局部变量,也没有传递任何参数,因此可以不用为栈分配空间。
; 如果你的函数需要使用局部变量或者传递参数,可以使用sub rsp, n指令将栈指针向下移动n个字节,以便为局部变量和参数分配空间。在函数返回之前,可以使用add rsp, n指令将栈指针恢复到原来的位置。
; 需要注意的是,栈的大小应该是16字节的倍数,以便保证栈的对齐。在你的例子中,由于没有定义任何局部变量和参数,因此可以不用为栈分配空间。
; 在x86-64架构下,寄存器的大小为8字节,因此每个参数需要使用一个8字节的寄存器来传递。前四个参数会被依次传递到寄存器RCX、RDX、R8和R9中,因此这四个参数需要使用4个8字节的寄存器来传递。
; 在汇编语言中,16进制数通常使用h作为后缀来表示。
981





