сборка x64 бит - PullRequest
       15

сборка x64 бит

1 голос
/ 17 февраля 2011

Я начал программирование ассемблера (nasm) не так давно. Теперь я сделал функцию C с реализацией сборки, которая печатает целое число. Я работал с использованием расширенных регистров, но когда я хочу написать его с регистрами x64 (rax, rbx, ..), моя реализация завершается неудачно. Кто-нибудь из вас видит то, что я пропустил?

main.c:

#include <stdio.h>

extern void printnum(int i);

int main(void)
{
        printnum(8);
        printnum(256);

        return 0;
}

32-битная версия:

; main.c: http://pastebin.com/f6wEvwTq
; nasm -f elf32 -o printnum.o printnum.asm
; gcc -o printnum printnum.o main.c -m32

section .data
    _nl db 0x0A
    nlLen equ $ - _nl

section .text
    global printnum


printnum:
        enter 0,0

        mov eax, [ebp+8]

        xor ebx, ebx
        xor ecx, ecx
        xor edx, edx
        push ebx
        mov ebx, 10

startLoop:

        idiv ebx
        add edx, 0x30

        push dx ; With an odd number of digits this will screw up the stack, but that's ok
                ; because we'll reset the stack at the end of this function anyway.
                ; Needs fixing though.
        inc ecx
        xor edx, edx

        cmp eax, 0
        jne startLoop

        push ecx
        imul ecx, 2

        mov edx, ecx

        mov eax, 4 ; Prints the string (from stack) to screen
        mov ebx, 1
        mov ecx, esp
        add ecx, 4
        int 80h

        mov eax, 4 ; Prints a new line
        mov ebx, 1
        mov ecx, _nl
        mov edx, nlLen
        int 80h

        pop eax ; returns the ammount of used characters

        leave
        ret

x64 версия:

; main.c : http://pastebin.com/f6wEvwTq
; nasm -f elf64 -o object/printnum.o printnum.asm
; gcc -o bin/printnum object/printnum.o main.c -m64

section .data
    _nl db 0x0A
    nlLen equ $ - _nl

section .text
    global printnum

printnum:
    enter 0, 0

    mov rax, [rbp + 8]  ; Get the function args from the stac
    xor rbx, rbx
    xor rcx, rcx
    xor rdx, rdx

    push rbx        ; The 0 byte of the string
    mov rbx, 10     ; Dividor

startLoop:
    idiv rbx        ; modulo is in rdx
    add rdx, 0x30

    push dx

    inc rcx         ; increase the loop variable
    xor rdx, rdx        ; resetting the modulo

    cmp rax, 0
    jne startLoop

    push rcx        ; push the counter on the stack
    imul rcx, 2

    mov rdx, rcx        ; string length

    mov rax, 4
    mov rbx, 1
    mov rcx, rsp        ; the string
    add rcx, 4
    int 0x80

    mov rax, 4
    mov rbx, 1
    mov rcx, _nl
    mov rdx, nlLen
    int 0x80

    pop rax
    leave

    ret         ; return to the C routine

Заранее спасибо!

Ответы [ 2 ]

4 голосов
/ 17 февраля 2011

Я думаю, что ваша проблема в том, что вы пытаетесь использовать 32-битные соглашения о вызовах в 64-битном режиме. Это не сработает, если вы не вызываете эти процедуры сборки из C. 64-битное соглашение о вызовах задокументировано здесь: http://www.x86 -64.org / Documentation / abi.pdf

Кроме того, не используйте системные вызовы с открытым кодом. Вызовите обертки в библиотеке C. Таким образом, errno устанавливается правильно, вы используете sysenter / syscall, вам не нужно иметь дело с различиями между обычным соглашением о вызовах и соглашением об аргументах системного вызова, и вы изолированы от определенные проблемы ABI низкого уровня. (Другая ваша проблема в том, что write - это системный вызов номер 1, а не 4 для Linux / x86-64.)

Редакция в стороне: в настоящее время есть две и только две причины написать что-либо в сборке:

  1. Вы пишете один из немногих оставшихся битов глубокой магии, которые не могут быть написаны только на C (хороший пример - смелость libffi )
  2. Вы вручную оптимизируете подпрограмму внутреннего цикла, которая была измерена как критичная для производительности, а компилятор C не выполняет достаточно хорошую работу.

В противном случае просто напишите, что написано на C. Ваши преемники будут вам благодарны.

РЕДАКТИРОВАТЬ: проверенные номера системных вызовов.

2 голосов
/ 17 февраля 2011

Я не уверен, связан ли этот ответ с проблемой, с которой вы сталкиваетесь (поскольку вы ничего не указали в отношении ошибки), но 64-битный код имеет другое соглашение о вызовах, чем 32-битный кодделает.Оба основных 64-разрядных Intel ABI (Windows & Linux / BSD / Mac OS) передают функциональные параметры в регистры, а не в стек.Кажется, ваша программа все еще ожидает их в стеке, что не является обычным способом.

Редактировать: Теперь, когда я вижу, что есть подпрограмма C main(), которая вызывает ваши функции, мойОтвет точно о проблеме, с которой вы столкнулись.

...