[완료] C에서 어셈블리 어떻게 사용하는지 알려주세요.

0
points

그냥 어셈관련된 헤더 하나 추가하고 main이나 만든 함수에
어셈브리 써주면 지가 알아서 컴파일되나요?
예전에 이런 식으로 c에서 어셈블리를 쓴다는 내용을
본거 같기는한데 정확한건 잘 모르겠습니다.
잘 아시는 분 있으시면 간단한 예제와 함께 설명 좀 부탁드리겠습니다.

인라인 어셈블리를 말씀하시는 건가요??

inline assembly는 컴파일러마다 다릅니다.

0
points

어셈블리어 블럭 지정부터 문법까지 컴파일러마다 다 다르기 때문에 먼저 사용할 컴파일러를 알려주셔야
답변이 될 것 같네요. x86용 gcc라면 위의 링크를 참조하시면 될 듯 합니다만... x86용 Visual C++나
Borland C++ Builder 등을 사용하시면 또 다릅니다. 컴파일러에 따라 inline assembly 지원이 안되는
경우도 있으니 참고하시기 바랍니다.

컴파일러까지는 생각

0
points

컴파일러까지는 생각 안해 봤습니다.
gcc는 위 참조하면 될꺼 같고요.
굳이 안 알려주셔도 되지만
Visual C++도 알려 주시면 고맙겠네요^^
일단 윗 분 답글 달아주신거 감사하고요.
어느 정도 알았으니 완료는 해놓아야 겠네요.

VC++에서는 문법은 Intel Assembly 문법을 써야 합니다.

0
points

gcc와 같은 AT&T 문법을 사용하지 않으니 operand의 순서에 주의해야 합니다.
그리고 C에서 사용하는 변수들을 어느 정도까지 어셈블리 코드 내에서 직접 쓸 수 있습니다.
(gcc처럼 parameter 처리를 하지는 않습니다)

예를들어 i586 이상의 VC++용 코드로 CPU의 TSC 값을 읽는 코드를 작성한다면 다음과 같이 할 수 있습니다.

#include <stdio.h>

int main(void)
{
    union _tsc
    {
        unsigned long lvalue[2];
        __int64 qvalue;
    } tsc;

    __asm
    {
        rdtsc
        mov tsc.lvalue[0], eax
        mov tsc.lvalue[4], edx
    }

    printf("TSC Value: %I64d\n", tsc.qvalue);

    return 0;
}

1. code tag를 써도 <> 표시가 정상적으로 되지 않는군요. 다시 수정했습니다.

2. 다시 확인해보니 assembly code에서 배열 index를 잘못 계산했더군요. 수정합니다.

정태영의 이미지
14606
points

gcc 에서도 c 변수를

0
points

gcc 에서도 c 변수를 파라미터 처리하지 않고 그냥 사용할 수 있습니다. :)

다만 Mac OS X 의 XCode 에 포함된 gcc 가 아니라면 vc 에서처럼 어셈블럭을 사용하는 것이 아니라 명령어 하나하나를 __asm(" ") 으로 감싸줘야 하죠.

--
오랫동안 꿈을 그리는 사람은 그 꿈을 닮아간다...

http://mytears.org ~(~_~)~ 나 한줄기 바람처럼..

가능하기는 합니다만... 매우 제한적이고 상황에 따라 다릅니다.

0
points

일단 전역변수의 경우 해당 변수 이름에 대한 offset으로 접근할 수 있습니다. (gcc가 동작하는 platform에 따라 assembly에서 사용하는 변수명 형태가 다른 것은 별개의 문제로 생각하기로 합니다) 반면 VC++에서 지원하는 방식은 구조체나 공용체의 멤버 변수 형태까지도 지원합니다. gcc에서는 inline assembly 내에서 멤버 변수를 직접 지정할 수 없습니다. 그리고 지역변수의 경우 변수 이름을 사용할 수 없습니다. 지역변수는 무조건 [ebp+offset] 형태가 되거나 레지스터로 최적화 된 상태가 되므로 변수명으로 접근할 수 없는 것입니다. 아래의 예를 직접 컴파일 해 보시기 바랍니다. (linux gcc에서 시험해 봤습니다. Windows용 MinGW에서는 inline assembly내에서 접근한 변수명 앞에 _를 붙여야 합니다.)

#include <stdio.h>
 
union _tsc
{
    unsigned long lvalue[2];
    unsigned long long qvalue;
};
 
union _tsc tsc;
 
int main(void)
{
    union _tsc tsc2;

    /* 전역 변수의 경우 변수명에 대한 offset으로만 접근할 수 있음 */ 
    asm(
        "rdtsc\n\t"
        "movl %%eax, tsc\n\t"
        "movl %%edx, tsc+4\n\t"
        ::);

    /* 아래의 코드는 -S 옵션으로 assembly code output을 얻을 수는 있으나 컴파일은 불가능함 */ 
    asm(
        "rdtsc\n\t"
        "movl %%eax, tsc2\n\t"
        "movl %%edx, tsc2+4\n\t"
        ::);
 
    printf("TSC Value: %llu\n", tsc.qvalue);
    printf("TSC Value: %llu\n", tsc2.qvalue);
 
    return 0;
}

결론적으로 할 수는 있지만 일반적으로 사용하라고 권장할만한 방법은 아닙니다.

댓글 보기 옵션

원하시는 댓글 전시 방법을 선택한 다음 "설정 저장"을 누르셔서 적용하십시오.