如果我在Windows VC ++中有以下代码:
DWORD somevar = 0x12345678;
_asm call dword ptr [somevar]
如何使用AT& T语法在GCC内联汇编中做同样的事情?
__asm__ __volatile__ (
"call dword ptr [%%edx]" : :
"d" (somevar)
);
我尝试过这样的事情,但它会产生“垃圾”错误......
然后我尝试将somevar
传递给某些寄存器,然后将其转换为dword
,ptr
等,但我无法使其工作。
更新:我找到了一些有用的东西,就好像我们在这种情况下必须使用括号而不是括号一样,我找到了lcall
来调用{ {1}}。但我仍然无法理解如何重现far
。
答案 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”)的意思和作用,否则你将以你无法想象的方式破坏你的代码。在你能回答“早期的咒语”之前,甚至不要考虑使用它。