13

我正在研究 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

我的问题是:

  1. 我在谷歌上搜索了 nwm,它只告诉我 c++ 使用它来分配内存,有人可以告诉我更多关于它的信息吗?它在一个库中吗?如果可能的话,我可以得到它的源代码吗?如何?
  2. 我对这种语法不太熟悉,它想做什么?
4

1 回答 1

19

*用于调用或跳转指令的 AT&T 汇编语法中的绝对地址之前。这意味着它将跳转到寄存器中包含的地址。另一种方法是相对跳转,它是相对于当前指令的。

来自GNUas手册

AT&T 绝对(相对于 PC 相对)跳转/调用操作数以 `*' 为前缀;它们在 Intel 语法中是不受限制的。

在您的代码中,调用寄存器中的地址是有意义的。调用pA->show()需要查找以查看要调用的正确函数是什么。这是因为它是 A 类的虚函数。

于 2013-09-30T06:59:35.823 回答