我用x86_64编写这段代码,编译运行良好,
__asm__("nop"
: "=eax"(foo)
: //"eax"(foo),"ebx"(bar)
: "eax"
);
但是当我尝试在其他机器(x86_32)中编译时,gcc触发错误:S
inline1.c:6: error: impossible constraint in _asm_
我无法修复它.. :S
海湾合作委员会版本 ::
未编译 :
tur@aso:~$ gcc -v
Using built-in specs.
Target: i486-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 4.4.3-4ubuntu5.1' -- with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable- languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --enable-shared --enable-multiarch -- enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --program-suffix=-4.4 --enable-nls --enable-clocale=gnu --enable-libstdcxx-debug --enable-plugin --enable-objc-gc --enable-targets=all --disable-werror --with-arch-32=i486 --with-tune=generic --enable-checking=release --build=i486-linux-gnu --host=i486-linux-gnu --target=i486-linux-gnu
螺纹型号:磅gcc 版本 4.4.3 (Ubuntu 4.4.3-4ubuntu5.1)
编译
git@srvcode:/home/git/code/asm$ gcc -v
Using built-in specs.
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/4.6.1/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.6.1- 9ubuntu3' --with-bugurl=file:///usr/share/doc/gcc-4.6/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++,go --prefix=/usr --program-suffix=-4.6 --enable-shared --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.6 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-plugin --enable-objc-gc --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
螺纹型号:磅gcc 版本 4.6.1 (Ubuntu/Linaro 4.6.1-9ubuntu3)
谢谢大家!!
阿图尔。
nop
是一个无操作数指令。如果在内联中没有使用任何操作数,则 gcc 无法为使用的操作数分配任何寄存器,这可能就是设置任何输入/输出寄存器失败的原因。但是,指令不会修改寄存器,所以我很好奇为什么您认为eax
需要保留。
其次,在旧版本的 gcc 中,寄存器不能同时在输入/输出列表和 clobber 列表中。输入/输出列表中的寄存器当然被自然地假定为"破坏"。您最好调整到该行为以实现向后兼容性。
此外,您应该记住,内联程序集中的注册操作数不是使用其实际名称指定的(clobber 列表除外)。而是使用描述寄存器的单个字母。因此,我们有eax
a
,b
ebx
,等等。您可能还想阅读GCC内联程序集HOWTO,这是有关该主题的信息的绝佳资源。
总而言之,这样的事情应该效果更好:
__asm__("nop"
: /* no output */
: /* no input */
: "%eax" /* eax is clobbered */
);