示例:
int main(void)
{
int x = 10, y;
asm ("movl %1, %%eax;"
"movl %%eax, %0;"
:"=r"(y) /* y is output operand */
:"r"(x) /* x is input operand */
:"%eax"); /* %eax is clobbered register */
}r(y)?%%之前使用eax?通常使用单一的%,对吗?发布于 2010-08-28 02:32:21
好的,这是gcc的内联汇编程序,它很强大,但是很难理解。
首先,% char是一个特殊的字符。它允许您定义寄存器和数字占位符(稍后使用mor)。不幸的是,%也被用作寄存器名的一部分(例如%EAX),所以在gcc内联汇编程序中,如果要命名寄存器,就必须使用2%的字符。
%0,%1和%2 (等)是占位符输入和输出操作数。这些都是在列表中定义的,后面跟着汇编程序字符串。在您的示例中,%0成为y的占位符,%1成为x的占位符。编译器将确保在执行asm-代码之前变量在输入操作数的寄存器中,并确保输出操作数将被写入输出操作数列表中指定的变量。
现在您应该知道r(y)是什么:它是一个输入操作数,它为变量y保留一个寄存器,并将其赋值给占位符%1 (因为它是内联汇编程序字符串之后列出的第二个操作数)。还有很多其他占位符类型。M允许您指定内存位置,如果我没有弄错,可以将其用作数字常量。你会发现他们都列在gcc的文件里。
然后是便宜货清单。这张清单很重要!它列出了在汇编程序代码中修改的所有寄存器、标志、内存位置等(例如示例中的EAX )。如果您弄错了,优化器将不知道已经修改了什么,而且很可能您的代码无法工作。
顺便说一句,你的例子几乎毫无意义。它只是将值X加载到寄存器中,并将此寄存器分配给EAX。之后,EAX将被存储到另一个寄存器中,然后这个寄存器将成为您的y变量。所以它所做的只是一个简单的任务:
Y= x;
最后一件事:如果你曾经使用过英特尔风格的汇编程序:你必须倒读这些论点。对于所有指令,源操作数是跟随指令本身的操作数,目标操作数是逗号右侧的操作数。与Intel语法相比,这正好相反。
发布于 2010-08-28 02:31:54
试试本教程。它涵盖了您所要求的一切:例如,尝试第6节 -它很好地解释了约束,以及"=“符号的含义。甚至还涵盖了破门而入的登记册的概念(第5.3节)。
发布于 2010-08-28 02:34:30
带有"r“或"=r”的行是操作数约束。"=“指输出操作数。实质上,这是:
:"=r"(y)
:"r"(x)意味着%0 (即:第一个操作数)对应于y并用于输出,%1 (第二个操作数)对应于x。
单个%通常用于AT&T语法程序集,但对于内联程序集,单%用于操作数引用(例如:%0,%1),而双%用于文字寄存器引用。想象一下,如果要在输出中使用文字%,就必须在printf格式中使用双%。
一个失败的寄存器是一个寄存器,它的值将被程序集代码修改。从代码中可以看到,eax是写到的。您需要告诉gcc这件事,这样它就知道编译后的代码不能在eax中保存它以后要调用这个程序集时所需的任何东西。
https://stackoverflow.com/questions/3589157
复制相似问题