我正在研究c ++如何通过汇编语言调用正确的成员函数。 我带来的简单程序如下:
class A
{
public:
virtual void show() {}
};
class B : public A
{
public:
void show() {}
};
int main()
{
A* pA = new B;
pA->show();
return 0;
}
其装配如下:
main:
.LFB2:
.cfi_startproc
.cfi_personality 0x3,__gxx_personality_v0
pushq %rbp
.cfi_def_cfa_offset 16
movq %rsp, %rbp
.cfi_offset 6, -16
.cfi_def_cfa_register 6
pushq %rbx
subq $24, %rsp
movl $8, %edi
.cfi_offset 3, -24
call _Znwm <<<================ 1
movq %rax, %rbx
movq %rbx, %rax
movq %rax, %rdi
call _ZN1BC1Ev
.L11:
movq %rbx, %rax
movq %rax, -24(%rbp)
movq -24(%rbp), %rax
movq (%rax), %rax
movq (%rax), %rdx
movq -24(%rbp), %rax
movq %rax, %rdi
call *%rdx <<<=== 2
movl $0, %eax
addq $24, %rsp
popq %rbx
leave
ret
.cfi_endproc
我的问题是:
答案 0 :(得分:14)
在呼叫或跳转指令的AT&amp; T汇编语法中的绝对地址之前使用*
。这意味着它将跳转到寄存器中包含的地址。另一种选择是相对跳跃,它相对于当前指令。
AT&amp; T绝对值(与PC相对) 跳转/调用操作数以“*”为前缀;他们是无限的 英特尔语法。
在你的代码中,调用寄存器中的地址是有意义的。调用pA->show()
需要查看要调用的函数是什么。这是因为它是A类的虚函数。