2014-09-03 4 views
3

어셈블리 언어를 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; 
} 

감사

업데이트이 프로그램은 GCC 4.8.3에 예상 된 결과를 제공하지만 GCC 4.6.3에있다. 나는 비 작동 코드의 어셈블리 출력 붙여 오전 : 당신은 두 번 x 말을 할 필요가 없습니다

.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 
+0

내 x86_64 리눅스에서 괜찮습니다. –

+0

@rakib gcc를 사용 했습니까? 단지 관련성이있는 경우 gcc 4.6.3을 사용하고 있습니다. – Arani

+1

gcc에서 생성하는 어셈블리 코드를 확인하십시오. 'gcc -S test.c'는 어셈블리리스트와 함께'test.s'를 만들 것입니다. – afenster

답변

4

을; 일단 충분한 :

asm("incl %0": "+r"(x)); 

+r

는 입력 값과 출력 것이라고 말한다.

입력 및 출력 레지스터가 각각 다른 방법으로 입력을 받아 %1에서 가져와 추가하고 %0에 출력을 작성해야하지만 incl으로 할 수는 없습니다.

일부 컴파일러에서 작동하는 이유는 GCC가 %0%1을 동일한 레지스터에 할당 할 수 있기 때문입니다. 그런 경우에는 그렇게 할 수 있지만 그렇게 할 필요는 없습니다. 덧붙여서, 을 쓰고 싶다면 GCC가 동일한 레지스터에 입출력을 할당하는 것을 막을 수 있습니다. 예를 들어 입력을 사용하여 최종 출력을 계산하기 전에 출력을 초기화하려면 & 수정자를 사용해야합니다.

수정 자에 대한 문서는 here입니다.