GCC内联汇编:调用dword ptr

时间:2010-08-10 04:23:34

标签: c++ linux gcc assembly inline-assembly

如果我在Windows VC ++中有以下代码:

DWORD somevar = 0x12345678;
_asm call dword ptr [somevar]

如何使用AT& T语法在GCC内联汇编中做同样的事情?

__asm__ __volatile__ (
    "call dword ptr [%%edx]" : :
    "d" (somevar)
);

我尝试过这样的事情,但它会产生“垃圾”错误......

然后我尝试将somevar传递给某些寄存器,然后将其转换为dwordptr等,但我无法使其工作。

更新:我找到了一些有用的东西,就好像我们在这种情况下必须使用括号而不是括号一样,我找到了lcall来调用{ {1}}。但我仍然无法理解如何重现far

1 个答案:

答案 0 :(得分:7)

您不使用带有AT& T汇编语法的DWORD PTR或类似内容。操作数长度通常取自寄存器名称(可以选择提供带助记符的后缀),而后者又来自您给asm()的C操作数的大小。这是内联汇编程序的一个非常好的属性,因为它意味着如果您为x86_64 i386体系结构编译它,此示例将运行。在第一种情况下,程序集变为call *%rdx,在第二种情况下变为call *%edx

#include <stdio.h>
void p()
{
    puts("Hallo");
}
typedef void (*fp)();
int main()
{
    fp var=&p;
    var();  
    // put 'var' in a register, the '*' says call indirect:
    asm volatile ("call *%0"::"r"(var));
}

您可以阅读GCC生成的代码(尝试使用-S进行编译)以了解AT&amp; T语法。也尝试谷歌一些介绍,如:

警告:使用gcc进行内联汇编并非易事。这是因为,除了大多数其他编译器之外,它还可以用优化器一起工作。你必须真正理解约束(“r”)的意思和作用,否则你将以你无法想象的方式破坏你的代码。在你能回答“早期的咒语”之前,甚至不要考虑使用它。