Ara*_*ani 5 c x86 assembly gcc
我试图了解如何在 C 中嵌入汇编语言(在 x86_64 架构上使用 gcc)。我编写这个程序是为了增加单个变量的值。但我得到的输出是垃圾值。和想法为什么?
#include <stdio.h>
int main(void) {
int x;
x = 4;
asm("incl %0": "=r"(x): "r0"(x));
printf("%d", x);
return 0;
}
Run Code Online (Sandbox Code Playgroud)
谢谢
更新该程序在 gcc 4.8.3 上给出了预期结果,但在 gcc 4.6.3 上没有给出预期结果。我粘贴非工作代码的汇编输出:
.file "abc.c"
.section .rodata
.LC0:
.string "%d"
.text
.globl main
.type main, @function
main:
.LFB0:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
pushq %rbx
subq $24, %rsp
movl $4, -20(%rbp)
movl -20(%rbp), %eax
incl %edx
movl %edx, %ebx
.cfi_offset 3, -24
movl %ebx, -20(%rbp)
movl $.LC0, %eax
movl -20(%rbp), %edx
movl %edx, %esi
movq %rax, %rdi
movl $0, %eax
call printf
movl $0, %eax
addq $24, %rsp
popq %rbx
popq %rbp
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE0:
.size main, .-main
.ident "GCC: (Ubuntu/Linaro 4.6.3-1ubuntu5) 4.6.3"
.section .note.GNU-stack,"",@progbits
Run Code Online (Sandbox Code Playgroud)
你不需要说x两次;一次就足够了:
asm("incl %0": "+r"(x));
Run Code Online (Sandbox Code Playgroud)
表示+r该值将被输入和输出。
您使用单独的输入和输出寄存器的方式要求您从 中获取输入%1,加一,然后将输出写入%0,但您不能使用 来做到这一点incl。
它在某些编译器上工作的原因是因为 GCC 可以自由地将%0和分配%1给同一个寄存器,并且在这些情况下似乎已经这样做了,但它不必这样做。顺便说一句,如果您想阻止GCC 将输入和输出分配到同一寄存器(例如,如果您想在使用输入计算最终输出之前初始化输出),则需要使用修饰符&。
修饰符的文档位于此处。