我花了最近两天时间研究AT& T内联装配,但我在转换这个装配时遇到了一些问题:
static char vendername[50] = {0};
_asm {
mov eax, 0
cpuid
mov dword ptr [vendername], ebx
mov dword ptr [vendername+4], edx
mov dword ptr [vendername+8], ecx
}
这是我的尝试:
static char vendername[50] = {0};
__asm__(
"movl $0,%%eax \n"
"cpuid \n"
"movl %%ebx, %[vendername] \n"
"movl %%edx, %[vendername+$4] \n"
"movl %%ecx, %[vendername+$8] \n"
:"=r"(vendername) //user vendername as output
:[vendername]"m"(vendername) //use vendername as input
:"eax","ebx","edx","ecx" // clobbers those regs
);
此代码不起作用,gcc给我以下错误:
error: undefined named operand ‘vendername+4’
和
error: undefined named operand ‘vendername+8’
如何将代码从英特尔组装转换为AT& T组件?
答案 0 :(得分:3)
经验法则是,如果你在内联asm中写mov
,你可能做错了:)
编译器可以为您自己加载/存储值,即
int dummy;
union {
char text[12];
struct {
int ebx;
int edx;
int ecx;
};
} vendorname;
__asm__(
"cpuid \n"
: "=b" (vendorname.ebx), "=d" (vendorname.edx), "=c" (vendorname.ecx), "=a" (dummy)
: "a" (0)
);
请注意,由于必须将3个双字解释为字符串,因此案例很复杂。
答案 1 :(得分:1)
你可以这样做:
#include <stdio.h>
int main(int argc, char **argv) {
static char vendername[50] = {0};
__asm__ __volatile__ (
"movl $0, %%eax\n"
"cpuid\n"
"movl %%ebx, %0\n"
"movl %%edx, %0 + 4\n"
"movl %%ecx, %0 + 8\n"
:"=m"(vendername)
:
:"eax", "ebx", "edx", "ecx"
);
printf("%s\n", vendername);
return 0;
}