使用GCC Extended ASM将内联Intel ASM转换为AT& T ASM

时间:2013-10-16 18:50:40

标签: gcc assembly x86 inline-assembly

我花了最近两天时间研究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组件?

2 个答案:

答案 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;
}